开启掘金成长之旅!这是我参与「掘金日新计划 · 2 月更文挑战」的第 30 天,点击查看活动详情
前言
今天是更文挑战的最后一天了,上一篇(安卓开发控件学习——使用 PreviewView预览相机画面 - 掘金 (juejin.cn))留了个坑,作者没把动态获取权限的代码附上,本篇我们就来具体聊聊它。
正文
背景与官方建议
早在安卓远古时期,也就是Android 6.0(API 级别 23)之前都是不需要应用运行时额外去获取权限的(安装之前向用户统一请求),也就是放清单文件注册权限即可,但后面Google出于用户隐私和安全性的考量,逐渐收紧权限的开放,有些权限都列为危险权限,只能去应用运行时动态获取。
于是就出现如果你不向用户申请权限(弹窗,用户选择),有些功能就无法正常运行,比如调用摄像头,所以我们通过动态获取权限的方式使功能可以正常使用(用户永拒时给予弹窗提示告知用户功能无法使用的原因),而Google当然也告诉了我们开发者如何去动态请求权限,我们可以在官方的文档看到相关教程(请求运行时权限 | Android 开发者 | Android Developers (google.cn))【中文版真的蛮友好的】:
参考用例(Kotlin): permissions-samples/RuntimePermissionsBasicKotlin at main · android/permissions-samples (github.com)
官方将请求的方法以及例子都写好了,如果想使用系统方法或者自己封装请求权限的话用官方的教程已经足够,但为了偷个懒,作者就用了Google写好的用于简易请求权限的封装包(googlesamples/easypermissions: Simplify Android M system permissions (github.com)),我们将依赖导入项目即可。
使用封装库请求
根据官网提示,我们先导入依赖:
dependencies {
...
implementation 'pub.devrel:easypermissions:3.0.0'
...
}
然后同步好Gradle后就能直接开始使用EasyPermissions去请求我们的权限了。
使用方法很简单,我们如果需要Camera权限,就先在Activity类名位置继承我们的EasyPermissions.PermissionCallbacks接口:
class MainActivity : AppCompatActivity() , EasyPermissions.PermissionCallbacks {
...
}
同时,我们也就需要实现接口里的所有方法,分别是onPermissionsGranted()和onPermissionsDenied()方法,用来监听同意或拒绝权限回调,如果需要对用户永久拒绝的权限做提示就可以放onPermissionsDenied()方法实现:
override fun onPermissionsGranted(requestCode: Int, perms: MutableList<String>) {
for (temp in perms) {
if (temp == android.Manifest.permission.CAMERA) {
Log.e(_TAG, "CAMERA onPermissionsGranted")
}
}
}
override fun onPermissionsDenied(requestCode: Int, perms: MutableList<String>) {
Log.d(_TAG, "onPermissionsDenied:" + requestCode + ":" + perms.size)
}
在我们的onPermissionsDenied()方法去实现对用户永久禁止相机权限提示到设置页去打开APP的相机权限,调用AppSettingsDialog方法弹窗提示:
override fun onPermissionsDenied(requestCode: Int, perms: MutableList<String>) {
Log.d(_TAG, "onPermissionsDenied:" + requestCode + ":" + perms.size)
val permissions = android.Manifest.permission.CAMERA
if (perms.contains(permissions)) {
if (EasyPermissions.somePermissionPermanentlyDenied(this, perms)) {
AppSettingsDialog.Builder(this)
.setTitle("The permission has been denied by you")
.setRationale("If you do not open the permission, you cannot use this function, click OK to open the permission")
.setRequestCode(12345)
.build()
.show()
}
}
然后我们去封装一个request()请求权限方法:
private fun request() {
val permissions = arrayOf(
android.Manifest.permission.CAMERA,
//可以添加其他的权限,用来判断
)
//判断有没有权限
if (EasyPermissions.hasPermissions(this, *permissions)) {
// 有权限,需要做什么
} else {
// 没有权限, 申请权限
EasyPermissions.requestPermissions(
this,
"摄像机需要用户允许才能调用,请开启相关权限(理由)",
1,
*permissions
)
}
}
我们使用EasyPermissions的hasPermissions方法去判断是否给了权限,如果没有权限,我们使用requestPermissions方法去提示用户开启权限。
这个request()方法我们直接放在我们需要调用相关需要权限的地方即可,比如上一篇的使用摄像机的代码前面:
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
binding = ActivityMainBinding.inflate(layoutInflater)
setContentView(binding.root)
...
// 判断是否要请求相机权限
request()
//请求CameraProvider
cameraProviderFuture = ProcessCameraProvider.getInstance(this)
cameraProviderFuture.addListener({
val cameraProvider = cameraProviderFuture.get()
bindPreview(cameraProvider)
}, ContextCompat.getMainExecutor(this))
}
至此,基本的动态权限请求就完成了,我们去运行程序来看看效果:
首先是发现用户没有给相机权限提示:
然后如果拒绝,下一次进来APP就会提示用户:
这个地方的逻辑注意,如果正常的话,应该放入拒绝的回调,这样用户拒绝时就能提示到用户。
最后是用户永久拒绝不提示时,我们的弹窗提示用户去设置权限的地方去开启:
总结
用户的隐私更加安全,但需求也变得更难做,可还是值得提倡,让用户的权利得到保障!