轻量级Android6.0动态权限解决方案

来源:互联网 发布:算量软件有哪些 编辑:程序博客网 时间:2024/06/03 21:34

在手机系统中,权限是一个非常重要的机制,它赋予了一个应用的权限范围,比如打电话的应用必须有电话权限,拍照的应用比较有拍照的权限。在Android6.0之前,Android采用的是静态权限机制,也就是我们将需要的权限(可能并不需要)全都写在manifest中,然后Android应用在安装的时候就必须授予这些权限。这种机制就产生了一个问题,一个打电话的应用可能也申请了拍照的权限,一个拍照的应用申请了读取短信内容的权限。这种现象对用户是非常不利的,一方面用户并不清楚应用到底需要什么权限,没有直观上的感觉;另一方面,这种一包揽的机制,导致应用申请了很多不需要的权限,其中一部分涉及到了用户的隐私,造成了用户隐私的泄漏。比如我们使用一款相机APP拍照的时候,刚打开相机,自己的短信内容就被上传到服务器了。。。
后来,Google意识到了这个问题,引入了动态权限机制。动态权限机制将权限分成了两部分,一部分是一般权限(Normal Permissons),一般不涉及用户的隐私,比如ACCESS_NETWORK_STATE,ACCESS_WIFI_STATE等;一部分是危险权限(Dangerous Permissons),这些权限涉及到了用户的隐私或其它比较重要的信息,比如WRITE_CONTACTS, CAMERA等。在处理权限时,首先我们还要像以前一样将需要的权限都写到manifest中;另一方面,针对危险权限我们应该在需要这些权限之前申请它们,系统在收到申请后,会提示用户是什么权限,请求用户授予或者拒绝。需要注意的是,Android6.0引入的权限组(permission group)的概念,对于在同一个权限组的权限,我们只要申请其中一个,就会将这个组的所有权限都授予给我们。

权限列表

Normal Permissions

ACCESS_LOCATION_EXTRA_COMMANDSACCESS_NETWORK_STATEACCESS_NOTIFICATION_POLICYACCESS_WIFI_STATEBLUETOOTHBLUETOOTH_ADMINBROADCAST_STICKYCHANGE_NETWORK_STATECHANGE_WIFI_MULTICAST_STATECHANGE_WIFI_STATEDISABLE_KEYGUARDEXPAND_STATUS_BARGET_PACKAGE_SIZEINSTALL_SHORTCUTINTERNETKILL_BACKGROUND_PROCESSESMODIFY_AUDIO_SETTINGSNFCREAD_SYNC_SETTINGSREAD_SYNC_STATSRECEIVE_BOOT_COMPLETEDREORDER_TASKSREQUEST_INSTALL_PACKAGESSET_ALARMSET_TIME_ZONESET_WALLPAPERSET_WALLPAPER_HINTSTRANSMIT_IRUNINSTALL_SHORTCUTUSE_FINGERPRINTVIBRATEWAKE_LOCKWRITE_SYNC_SETTINGS

Dangerous Permissions:

group:android.permission-group.CONTACTS  permission:android.permission.WRITE_CONTACTS  permission:android.permission.GET_ACCOUNTS  permission:android.permission.READ_CONTACTSgroup:android.permission-group.PHONE  permission:android.permission.READ_CALL_LOG  permission:android.permission.READ_PHONE_STATE  permission:android.permission.CALL_PHONE  permission:android.permission.WRITE_CALL_LOG  permission:android.permission.USE_SIP  permission:android.permission.PROCESS_OUTGOING_CALLS  permission:com.android.voicemail.permission.ADD_VOICEMAILgroup:android.permission-group.CALENDAR  permission:android.permission.READ_CALENDAR  permission:android.permission.WRITE_CALENDARgroup:android.permission-group.CAMERA  permission:android.permission.CAMERAgroup:android.permission-group.SENSORS  permission:android.permission.BODY_SENSORSgroup:android.permission-group.LOCATION  permission:android.permission.ACCESS_FINE_LOCATION  permission:android.permission.ACCESS_COARSE_LOCATIONgroup:android.permission-group.STORAGE  permission:android.permission.READ_EXTERNAL_STORAGE  permission:android.permission.WRITE_EXTERNAL_STORAGEgroup:android.permission-group.MICROPHONE  permission:android.permission.RECORD_AUDIOgroup:android.permission-group.SMS  permission:android.permission.READ_SMS  permission:android.permission.RECEIVE_WAP_PUSH  permission:android.permission.RECEIVE_MMS  permission:android.permission.RECEIVE_SMS  permission:android.permission.SEND_SMS  permission:android.permission.READ_CELL_BROADCASTS

动态权限申请过程

由于动态权限机制是在Android6.0引入的,所有为了适配系统,Google给我们提供了ActivityCompat和ContextCompat,在这两个类的权限相关的方法中都为我们做了系统版本适配,所以就不需要我们自己适配了。

动态权限的申请一般分成两种情况,
第一种情况是没有点击”不在显示”按钮,这个时候申请流程是这样的:
检查权限(checkSelfPermission),检查通过则执行相关业务代码,检查不通过则申请权限(requestPermissions),系统收到申请后提示用户授予权限,然后回调请求权限结果方法(onRequestPermissionsResult),在这个回调中如果用户授予了申请的权限,则执行相关业务代码,拒绝了则提示用户手动设置权限,否则应用无法正常运行。
第二种情况是在系统在提示用户授予权限的时候,点击了“不在显示”按钮,这个情况下,用户只能选择拒绝,并且以后启动应用的时候系统会直接调用onRequestPermissionsResult方法,并且传递权限申请结果为拒绝(PackageManager.PERMISSION_DENIED),这个时候我们应该提醒用户手动设置权限,否则应用无法正常运行。实际上,在国内很多系统已经将“不在显示”按钮去掉了(比如小米系统),这个时候如果我们拒绝了权限,当我们下次打开应用的时候,系统也是会直接执行onRequestPermissionsResult,并且回调结果为拒绝(PackageManager.PERMISSION_DENIED),所以说这个时候应该等同于点击了“不在显示”按钮了,需要我们提示用户手动授予权限。

动态权限申请代码实现

动态权限的申请虽然代码量不大,也没什么难度,但是由于代码比较分散,而且套路都是一样的,有比较多的重复代码,所以我对权限申请进行了简单的封装。
代码量很少,只有两个类IRequestPermissionCallback和ZPermissionHelper。
其中IRequestPermissionCallback是一个回调接口,需要申请权限的activity需要实现这个接口,并在相应的接口中实现自己的业务操作。代码如下;

public interface IRequestPermissionCallback { // 权限授予失败的回调默认为调起系统权限设置activity    /**     * 检查权限时,已经授予了所有权限     */    void onPermissionGranted();    /**     * 权限授予成功回调     * @param requestCode     * @param permissions     * @param grantResults     */    void onAfterPermissionGranted(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults);    /**     * 权限被拒绝后调起(如果选择不再显示,则不会被调用)     * @param activity     * @param permission     */    void onShouldShowRequestPermissionRationale(@NonNull Activity activity, @NonNull String permission);}

在这个接口中,我设计了三个方法,
onPermissionGranted:当我们最开始的时候检测权限的时候,如果通过了,我们就可以执行自己的业务代码了,所以把这个抽象出来,就可以有不同的代码实现了。
onAfterPermissionGranted:当我们申请权限,并被成功授予权限后的回调。
本来还有一个方法是当我们申请权限,被拒绝后的回调,但是我后来直接把它写成跳转到权限设置界面了。
onShouldShowRequestPermissionRationale:当权限申请第一次被拒绝以后,再次申请的时候onShouldShowRequestPermissionRationale就会返回ture,这个时候我们可以加一个Toast,提醒用户权限的重要性。

以上是回调接口设计,ZPermissionHelper则封装了权限申请的主要代码,也很简单,代码如下;

public class ZPermissionHelper {    private Activity mContext;    final IRequestPermissionCallback mCallback;    public static final int ACTION_APPLICATION_DETAILS_SETTINGS = 0X101;    public static final String PACKAGE_URL_SCHEME = "package:";//权限方案    private int PERMISSION_GRANTED = PackageManager.PERMISSION_GRANTED;    private int PERMISSION_DENIED = PackageManager.PERMISSION_DENIED;    private String[] REQUEST_PERMISSIONS;    private int REQUEST_PERMISSIONS_TAG;    public ZPermissionHelper(Activity mContext, String[] REQUEST_PERMISSIONS, int requestPermissionTag,                             IRequestPermissionCallback callback) {        this.mContext = mContext;        this.REQUEST_PERMISSIONS = REQUEST_PERMISSIONS;        REQUEST_PERMISSIONS_TAG = requestPermissionTag;        mCallback = callback;    }    /**     * 检查权限     */    public void checkPermissions() {//        if (!isCheckPermission) return;        if (checkSelfPermisson()) {            // 权限被授予            mCallback.onPermissionGranted();        }    }    private boolean checkSelfPermisson() {        for (int i = 0; i < REQUEST_PERMISSIONS.length; i++) {            String permission = REQUEST_PERMISSIONS[i];            if (isLeakPermission(permission)) {                ActivityCompat.requestPermissions(mContext, REQUEST_PERMISSIONS, REQUEST_PERMISSIONS_TAG);                return false;            }        }        return true;    }    /**     * 检查具体权限缺失     * @param permission     * @return     */    private boolean isLeakPermission(String permission) {        int selfPermission = ActivityCompat.checkSelfPermission(mContext, permission);        if (selfPermission == PERMISSION_DENIED) {            if (ActivityCompat.shouldShowRequestPermissionRationale(mContext, permission)) {                // 显示这个权限的重要性,给用户提示                mCallback.onShouldShowRequestPermissionRationale( mContext, permission);            }            return true;        }        return false;    }    /**     * 权限请求结果回调     * @param requestCode     * @param permissions     * @param grantResults     */    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {        if (requestCode == REQUEST_PERMISSIONS_TAG) {            if (permissions.length > 0 && hasAllPermissionGranted(grantResults)) {                // 授予权限成功                mCallback.onAfterPermissionGranted(requestCode, permissions, grantResults);            } else {                showMissingPermissionDialog();                // 授予权限失败            }        }    }    /**     * 申请的权限是否全部授予     * @param grantResults     * @return     */    public boolean hasAllPermissionGranted(int[] grantResults) {        for (int grantResult : grantResults) {            if (grantResult == PERMISSION_DENIED) {                return false;            }        }        return true;    }    /**     * 显示提示对话框     */    public void showMissingPermissionDialog() {        AlertDialog.Builder builder = new AlertDialog.Builder(mContext);        builder.setTitle("帮助");//提示帮助        builder.setMessage("当前应用缺少必要权限。\n请点击\"设置\"-\"权限\"-打开所需权限。\n最后点击两次后退按钮,即可返回。");        builder.setPositiveButton("设置", new DialogInterface.OnClickListener() {            @Override            public void onClick(DialogInterface dialog, int which) {                startPermissionGrantActivity();            }        });        builder.setNegativeButton("退出", new DialogInterface.OnClickListener() {            @Override            public void onClick(DialogInterface dialog, int which) {                // 退出应用                mContext.finish();            }        });        builder.setCancelable(false);        builder.show();    }    /**     * 调起系统权限控制界面     */    private void startPermissionGrantActivity() {        Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS);        intent.setData(Uri.parse(PACKAGE_URL_SCHEME + mContext.getPackageName()));//        mContext.startActivity(intent);        mContext.startActivityForResult(intent, ACTION_APPLICATION_DETAILS_SETTINGS);    }}

首先是构造函数;

public ZPermissionHelper(Activity mContext, String[] REQUEST_PERMISSIONS, int requestPermissionTag,                             IRequestPermissionCallback callback) {        this.mContext = mContext;        this.REQUEST_PERMISSIONS = REQUEST_PERMISSIONS;        REQUEST_PERMISSIONS_TAG = requestPermissionTag;        mCallback = callback;    }

构造函数中接收activity的实例,REQUEST_PERMISSIONS是需要申请权限的String数组,requestPermissionTag是申请权限时用到的requestCode,用于在onRequestPermissionsResult中接收回调结果。
然后是检查权限;

 /**     * 检查权限     */    public void checkPermissions() {//        if (!isCheckPermission) return;        if (checkSelfPermisson()) {            // 权限被授予            mCallback.onPermissionGranted();        }    }    private boolean checkSelfPermisson() {        for (int i = 0; i < REQUEST_PERMISSIONS.length; i++) {            String permission = REQUEST_PERMISSIONS[i];            if (isLeakPermission(permission)) {                ActivityCompat.requestPermissions(mContext, REQUEST_PERMISSIONS, REQUEST_PERMISSIONS_TAG);                return false;            }        }        return true;    }/**     * 检查具体权限缺失     * @param permission     * @return     */    private boolean isLeakPermission(String permission) {        int selfPermission = ActivityCompat.checkSelfPermission(mContext, permission);        if (selfPermission == PERMISSION_DENIED) {            if (ActivityCompat.shouldShowRequestPermissionRationale(mContext, permission)) {                // 显示这个权限的重要性,给用户提示                mCallback.onShouldShowRequestPermissionRationale( mContext, permission);            }            return true;        }        return false;    }

遍历权限数组,检查每个权限是否被授予,如果权限检查通过了,就直接回调onPermissionGranted方法,实现业务逻辑;只要有一个权限没有被授予就立马停止遍历,然后申请权限。
最后处理申请回调方法:

public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {        if (requestCode == REQUEST_PERMISSIONS_TAG) {            if (permissions.length > 0 && hasAllPermissionGranted(grantResults)) {                // 授予权限成功                mCallback.onAfterPermissionGranted(requestCode, permissions, grantResults);            } else {                showMissingPermissionDialog();                // 授予权限失败            }        }    }

这个方法是我们自己的onRequestPermissionsResult,在activity的onRequestPermissionsResult中调用,将处理代码封装到我们的ZPermissionHelper中。
在这里首先判断requestCode, hasAllPermissionGranted(grantResults)判断申请的权限是否被全部授予,

  public boolean hasAllPermissionGranted(int[] grantResults) {        for (int grantResult : grantResults) {            if (grantResult == PERMISSION_DENIED) {                return false;            }        }        return true;    }

如果成功授予,则回调onAfterPermissionGranted方法,否则就直接显示对话框,提醒用户手动设置权限。
以上就是ZPermissionHelper的代码,非常简单,非常轻量。
以下是调用ZPermissionHelper的Activity的代码

public class MainActivity extends AppCompatActivity implements IRequestPermissionCallback {    private static final int PERMISSION_REQUEST = 0x101;    private String[] REQUEST_PERMISSIONS = new String[]{            Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.CAMERA    };    private ZPermissionHelper mPermissionHelper;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        mPermissionHelper = new ZPermissionHelper(this, REQUEST_PERMISSIONS, PERMISSION_REQUEST, this);        mPermissionHelper.checkPermissions();    }    @Override    public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {        super.onRequestPermissionsResult(requestCode, permissions, grantResults);        mPermissionHelper.onRequestPermissionsResult(requestCode, permissions, grantResults);    }    @Override    protected void onActivityResult(int requestCode, int resultCode, Intent data) {        super.onActivityResult(requestCode, resultCode, data);        if (requestCode == ZPermissionHelper.ACTION_APPLICATION_DETAILS_SETTINGS) {            mPermissionHelper.checkPermissions();        }    }    @Override    public void onPermissionGranted() {        showToast("onPermissionGranted");    }    @Override    public void onAfterPermissionGranted(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {        showToast("onAfterPermissionGranted");    }    @Override    public void onShouldShowRequestPermissionRationale(@NonNull Activity activity, @NonNull String permission) {        showToast("onShouldShowRequestPermissionRationale");    }    public void showToast(String content) {        Toast.makeText(this, content, Toast.LENGTH_SHORT).show();    }}
  1. Activity实现IRequestPermissionCallback接口,并实现其方法
  2. 在onCreate方法中实例化ZPermissionHelper,并开启权限检查
  3. 在onRequestPermissionsResult回调中,将处理转移到ZPermissionHelper
    基本就是以上步骤,另外需要注意一点,如果我们拒绝了权限,我们应该通过startActivityForResult的方式打开权限设置界面,这样当从权限设置界面返回的时候,我们就可以立马开始权限检查,判断用户是不是已经将所需要的权限都开启了。

以上就是本篇的全部内容,在这个方案中,主要对权限检测和处理的代码进行了简单的封装,还有很多需要改进的地方,比如activity必须实现回调接口,并且看上去在activity中增加的代码也不少,希望大家提示宝贵的意见。

最后,建议将权限检测的代码放到BaseActivity中,这样就可以进一步的减少代码量了。:github地址

原创粉丝点击