动态权限
背景
从Android6.0版本开始google将权限分为普通权限和特殊权限,app必须在AndroidManifest.xml添加引用权限的语句。 在安装apk时安卓会将普通权限授予该app,但特殊权限需要运行时申请。
安卓按照权限类别分为权限组和权限, 每个权限都隶属于一个权限组。 当安卓系统授权一个权限时, 那么该权限所属权限组的权限都会自动被授权。
目前如果app的targetSdkVersion等于21,即按照Android5.0版本特性运行。 技术层面与市场上主流app差距较大, 功能层面也有一些功能可能失效(例如在一些机型上无法打电话、读写SD卡), 根本原因是没适配动态权限。
如何申请动态权限
判断当前手机系统是Android6.0及以上版本, 在Activity/Fragment里申请权限并处理权限结果回调。 这里要说明一下:Fragment是通过Activity申请权限的, 且权限回调onRequestPermissionResult也是Activity调用的Fragment该方法. 上图是权限申请流程图, 我们看到的权限弹窗对应/packages/apps/PackageInstaller/src/com/android/packageinstaller/permission/ui/GrantPermisssionsActivity.java, 点击“同意”或“不同意”通过PackageManager、AppOpsManager将权限操作更新到PackageManagerService和AppOpsService中。
调用Activity的申请权限方法其实是打开一个系统的Activity,操作结果通过setResult返回过来。
能不能直接调用PackageManager/AppOpsManagerd的方法授权给自己? 显然是不行的, PackageManagerService只允许在AndroidManifest.xml配置coreApp=”true”的应用修改权限,而普通app无法设置coreApp属性。
1 | public int getPermissionFlags(String name, String packageName, int userId) { |
如何判断权限
如上图所示,判断是否有权限最终会执行到PackagerManagerService的checkUidPermission函数中。
适配动态权限的方式
- 基本用法:在Activity、Fragment派生类中添加权限申请和结果回调。 坑:1、在插件中调用View的getContext返回值是PluginContext, 无法通过类型强转调用其附着Activity/Fragment的方法。2、如果界面层级很深,需要逐层添加回调参数。
- AOP方式,推荐https://github.com/permissions-dispatcher/PermissionsDispatcher, 在需要权限的函数上添加注解并在构建阶段注入代码。缺点是app插件中有多View控件如BaseCard无法使用。
- 第三方库https://github.com/yanzhenjie/AndPermission, 原理:新启动个透明Activity申请权限并保存回调函数到静态变量里,用户操作权限提示框结束后通过回调执行成功、失败逻辑。
示例代码: 为了避免适配动态权限逻辑产生风险, 可以新增if代码块做动态权限逻辑, else分支仍然是现有逻辑。 各业务线可能无法在同一个版本上搞定, 可以按照这种写法先后完成动态权限适配工作,待所有业务线都完成后调整宿主targetSdkVersion到26。
1 | if (getBaseContext().getApplicationInfo().targetSdkVersion >= Build.VERSION_CODES.M |
注意事项
- 适配小米机型动态权限;
- Android7.0版本开始使用FileProvider, 需要适配拍照功能;
- 适配DownloadManager安装apk;
- 用户禁用权限且不再提醒, 需要有个提示框提示用户去应用详情界面里放开权限, 弹窗建议使用CustomDialog(各业务UI样式统一)。
- 适配WindowManager悬浮窗;
附录
AndPermission库解决方案
1 | /** |