Android 权限申请分享

要求:

1.不用三方框架,减小安装包的大小

2.请求权限的时候,显示请求内容,配合应用市场上架要求

代码实现

调用的代码

kotlin 复制代码
//初始化
private val readLauncher = PermissionInfoLauncher(Manifest.permission.READ_CONTACTS) {
    "读取联系人".toast()
}

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    installSplashScreen()
    setContentView(binding.root)

    //注册
    lifecycle.addObserver(readLauncher)
    binding.btnReadContacts.setOnClickListener {
        //具体调用
        readLauncher.request(this)
    }
}

override fun onDestroy() {
    super.onDestroy()
    //解除绑定
    lifecycle.removeObserver(readLauncher)
}

核心实现类,其中 PermissionInfoDialog 、SettingPermissionDialog 是显示信息的弹框,跟设置的弹框,可以根据项目UI要求自行实现

注意Launcher 绑定,不能在OnCreate 中默认调用request,那时候Launcher还未初始化 如果遇到地图定位,在修改权限之后,还要重新自动定位,再加入

ini 复制代码
setLauncher = owner.registerForActivityResult(ActivityResultContracts.StartActivityForResult())

回调即可

kotlin 复制代码
class PermissionInfoLauncher(private val permission: String, private val function: () -> Unit) :
    DefaultLifecycleObserver {

    private lateinit var permissionLauncher: ActivityResultLauncher<String>

    @StringRes
    var infoRes: Int = R.string.permission_refuse_info_default

    @StringRes
    var setRes: Int = R.string.permission_set_default

    private val infoDialog: PermissionInfoDialog by lazy {
        PermissionInfoDialog.Builder().apply {
            content = CoreUtils.getApp().getString(infoRes)
        }.create()
    }

    override fun onCreate(owner: LifecycleOwner) {
        super.onCreate(owner)
        when (owner) {
            is FragmentActivity -> {
                permissionLauncher = owner.registerForActivityResult(ActivityResultContracts.RequestPermission()) { granted ->
                    infoDialog.dismiss()
                    if (granted) {
                        //处理业务逻辑
                        function()
                    } else {
                        val settingDialog = SettingPermissionDialog.Builder().apply {
                            content = owner.resources.getString(setRes)
                        }.create()
                        settingDialog.show(owner.supportFragmentManager)
                    }
                }
            }

            else -> {
                throw NullPointerException("${this.javaClass.name}观察的生命周期不是Activity或者Fragment")
            }
        }
    }

    override fun onDestroy(owner: LifecycleOwner) {
        super.onDestroy(owner)
        permissionLauncher.unregister()
    }

    fun request(activity: FragmentActivity) {
        infoDialog.show(activity.supportFragmentManager)
        if (::permissionLauncher.isInitialized) {
            permissionLauncher.launch(permission)
        }
    }
}
相关推荐
_Shirley1 小时前
鸿蒙设置app更新跳转华为市场
android·华为·kotlin·harmonyos·鸿蒙
hedalei3 小时前
RK3576 Android14编译OTA包提示java.lang.UnsupportedClassVersionError问题
android·android14·rk3576
锋风Fengfeng3 小时前
安卓多渠道apk配置不同签名
android
枫_feng3 小时前
AOSP开发环境配置
android·安卓
叶羽西4 小时前
Android Studio打开一个外部的Android app程序
android·ide·android studio
qq_171538855 小时前
利用Spring Cloud Gateway Predicate优化微服务路由策略
android·javascript·微服务
Vincent(朱志强)6 小时前
设计模式详解(十二):单例模式——Singleton
android·单例模式·设计模式
mmsx7 小时前
android 登录界面编写
android·登录界面
姜毛毛-JYM7 小时前
【JetPack】Navigation知识点总结
android
花生糖@8 小时前
Android XR 应用程序开发 | 从 Unity 6 开发准备到应用程序构建的步骤
android·unity·xr·android xr