android权限适配 简书,Android动态权限适配

最近APP里面要添加动态权限,网上找了不少例子并经过一定的测试,基本流程有了一定的认识,需要注意的地方记录下来以备以后查阅。

一 各大手机厂商的适配

依据各大手机Rom的不同表现,可以分为三种不同的模式:标准模式,默认开启模式,特殊权限模式。

1 标准模式:代表华为Mate8,三星S6。

基本流程如下:

用户启动APP后,第一次需要权限的时候,判断用户当前权限,弹出系统默认的系统Snacbarintstate = ActivityCompat.checkSelfPermission(activity,requestPermission);

通过onRequestPermissionsResult判断出来用户选择结果,

用户选择始终允许,则直接获取了权限,APP继续后续流程。

如果用户选择了拒绝,则需要APP自己处理,一般会弹出一个Dialog 提示用户需要该权限,是否去设置里面打开该权限。拒绝也分为两种情况,一种是选择了禁止后不再询问,一种是没有选择。两种的区别在于在标准模式下,用户选择了禁止不再提问后,不会再弹出系统Snacbar,而是直接返回拒绝的结果,如果用户没有选择,则会继续弹出系统Snacbar。

用户启动权限申请后,会有shouldShowRequestPermissionRationale来判断是否需要APP弹出一个提示框,提示用户来同意申请该动态权限。这个在应用首次申请权限时,如果用户点击拒绝,下次再申请权限,Android允许你提示用户,你为什么需要这个权限,更好引导用户是否授权。

本来是添加一个自定义Dialog来提示用户,后产品认为提示过多,则默认还是直接去申请权限。if(ActivityCompat.shouldShowRequestPermissionRationale(activity,requestPermission)) {

//为了优化用户关闭权限后 第一次没有任何提示

if(ConfigUtils.isHtc()) {

openSettingActivity(activity,activity.getString(R.string.permission_setting_head)

+ message +"权限",permissionGrant);

}else{

//TODO 直接进行权限申请 不再先进行自定义提示

ActivityCompat.requestPermissions(activity, newString[]{requestPermission},requestCode);

//                shouldShowRationale(activity, requestCode, requestPermission, permissionGrant);

}

}else{

ActivityCompat.requestPermissions(activity, newString[]{requestPermission},requestCode);

}

2 默认开启模式:代表HTC One。

和标准模式的区别,是调用检查APP权限的代码后,默认会给用户该权限。

所以不会弹出系统Snacbar提示你去进行选择,用户没有感觉到安装了一个动态权限版本的APP。

3 特殊权限模式:代表360手机。

360正常是属于默认开启模式的,但是针对特殊权限里面的部分权限,360认为是危险权限会进行自定义的提示。比如:照相机被认为是危险权限,打开本地存储则不属于。

针对危险权限的权限状态,返回的也是通过,但是360会自动打开自定义的Dialog,这个和系统默认的Snacbar是一个逻辑流程。

二  多权限处理

前提也是通过遍历,获取多个权限当前的状态,依据各自的权限状态再去分别处理。

多权限的返回结果:

比如 申请了三个权限A,B,C。A选择通过,B拒绝,C拒绝并不再提示,则在处理返回值的时候,需要分别处理,正常会弹出一个Dialog提示,B和C没有权限,需要到设置界面去打开。

等用户再次申请这三个权限的时候,肯定返回B和C需要申请,默认标准模式下会弹出系统Snacbar提示B权限的申请,而没有A和C,C已经拒绝提示,A已经通过。如果B通过了,则返回结果里面只有C没有通过,APP依然会弹出需要到设置界面去打开C权限的Dialog。

三 权限状态不准确的处理 代表HTC One

HTC属于默认申请就给权限的手机,但是用户在设置界面关闭了赋予APP的动态权限后,利用intstate = ActivityCompat.checkSelfPermission(activity,requestPermission);

得到的仍在是已经有权限的结果,这样会造成APP的实际没有权限,而后续的使用权限的操作会导致APP崩溃。针对这种情况需要添加AppOps来判断当前的权限状况:intstate = ActivityCompat.checkSelfPermission(activity,requestPermission);

if(state == PackageManager.PERMISSION_GRANTED) {

// 避免此时是已经被拒绝造成的通过

String appPermission =chageToAppOpsManagerC(requestCode);

if(null!= appPermission) {

checkResult =getPermissionState(activity,appPermission);

}

}private static int getPermissionState(Context activity,String  op) {

if(Build.VERSION.SDK_INT>= Build.VERSION_CODES.KITKAT) {

AppOpsManager appOpsManager = (AppOpsManager)                     activity.getSystemService(Context.APP_OPS_SERVICE);

returnappOpsManager.checkOpNoThrow(op,Binder.getCallingUid(),activity.getPackageName());

}

returnAppOpsManager.MODE_ALLOWED;

}

这时的返回值才是正确的状态。

备注:

动态权限优化了流程,原则就是只要有能弹出系统的Snacbar则就先弹Snacbar,不能弹出任何Snacbar之后,依据返回的结果APP给出提示需要什么权限,要跳转到系统设置里面赋予权限。

其他注意点

1 需要判断系统版本是否是6.0以上。低于6.0版本的不需要进行动态权限的判断,如果进入了动态权限的判断逻辑,则各种效果都会出来。

2 升级到6.0以上的版本编译之后,一些操作有了限制。比如,7.0之后对文件共享进行来限制,打开照相机,获取相机图片需要使用FileProvider进行操作。if(Build.VERSION.SDK_INT>= Build.VERSION_CODES.N) {//7.0及以上

Uri contentUri =getUriForFile(mContext,"包名.fileProvider",

newFile(FileUtils.getSDCardRoot() + ConfigFile.CACHE_AVATAR_DIR,ConfigFile.CACHE_AVATAR_CAMERA_NAME));

intentFromCapture.putExtra(MediaStore.EXTRA_OUTPUT,contentUri);

intentFromCapture.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION

| Intent.FLAG_GRANT_WRITE_URI_PERMISSION);

}else{

intentFromCapture.putExtra(MediaStore.EXTRA_OUTPUT,Uri

.fromFile(newFile(FileUtils.getSDCardRoot() + ConfigFile.CACHE_AVATAR_DIR,ConfigFile.CACHE_AVATAR_CAMERA_NAME)));

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值