Android AOP之Aspect-动态权限申请

PermissionJ

使用Aspect实现的面向切面进行Android动态权限申请
Github

使用:

  1. 根build.gradle
dependencies {
        classpath "com.hujiang.aspectjx:gradle-android-plugin-aspectjx:2.0.10"
}
  1. 需要申请权限的module build.gradle
plugins {
    id 'kotlin-kapt'
}

// 选配
aspectjx {
    // 需要织入代码的包名
    include 'com.xxx'
    // 不需要织入代码的包名
    exclude 'com.xxx'
    // 关闭AspectJX功能 enabled默认为true,即默认AspectJX生效
    enabled true
}

dependencies {

    implementation 'com.github.Archer1347:PermissionJ:1.0.0'
}
  1. 申请权限
@PermissionRequest(
        permissions = [Manifest.permission.WRITE_EXTERNAL_STORAGE, Manifest.permission.READ_EXTERNAL_STORAGE, Manifest.permission.CAMERA],
        requestCode = 1
    )
fun test() {
    Toast.makeText(this, "申请权限成功", Toast.LENGTH_SHORT).show()
}
  1. 申请权限失败(可选)
@PermissionRequestFailed
fun failed(permissionDetail: PermissionDetail) {
     Toast.makeText(this, "申请权限失败", Toast.LENGTH_LONG).show()
}

@PermissionRequestFailed
fun failed(permissionDetail: PermissionDetail) {
        Toast.makeText(
            this, "申请权限失败\n" +
                    "请求码:${permissionDetail.requestCode}\n" +
                    "成功:${permissionDetail.grantedPermissions?.joinToString()}\n" +
                    "失败:${permissionDetail.deniedPermissions?.joinToString()}\n" +
                    "是否勾选了不再提示:${permissionDetail.rejectRemind}", Toast.LENGTH_LONG
        ).show()
}

原理:

  1. 使用registerForActivityResult进行权限申请
  2. aspect织入代码时,通过获取上下文activity,添加一个空的Fragment用于权限请求
  3. 兜底:如果获取不到activity,则打开一个透明Activity用于添加fragment
@Aspect
@DelicateCoroutinesApi
class PermissionAspect {

    @Pointcut("execution(@com.permission.core.annotation.PermissionRequest * *(..))" + " && @annotation(permissionRequest)")
    fun requestPermissionMethod(permissionRequest: PermissionRequest) {
    }

    @Around("requestPermissionMethod(permissionRequest)")
    fun aroundJoinPoint(joinPoint: ProceedingJoinPoint, permissionRequest: PermissionRequest) {
        if (hasSelfPermissions(application, *permissionRequest.permissions)) {
            try {
                joinPoint.proceed()
            } catch (throwable: Throwable) {
                Log.d(TAG, throwable.localizedMessage.orEmpty())
            }
            return
        }
        requestPermissions(joinPoint, permissionRequest)
    }

    /**
     * Desc: 申请权限
     */
    private fun requestPermissions(joinPoint: ProceedingJoinPoint, permissionRequest: PermissionRequest) {
        GlobalScope.launch(Dispatchers.Main.immediate) {
            // 优先获取当前activity
            var activity = findActivityFromContext(joinPoint.`this`)
            // 如果获取不到当前activity,则启动一个透明Activity
            if (activity == null) {
                // 启动activity,onCreate之后返回activity实例
                activity = awaitStartActivityAndCreate(application, PermissionRequestActivity::class.java)
            }
            try {
                // 申请权限,返回申请结果
                val result = activity.requestPermissionsForResult(permissionRequest.permissions as Array<String>, permissionRequest.requestCode)
                if (activity is PermissionRequestActivity) {
                    activity.finish()
                }
                // 如果没有权限被拒绝,则权限申请通过
                if (result.deniedPermissions.isEmpty()) {
                    joinPoint.proceed()
                } else {
                    onDenied(joinPoint, result)
                }
            } catch (throwable: Throwable) {
                Log.e(TAG, throwable.localizedMessage.orEmpty())
            }
        }
    }

    /**
     * Desc: 权限被拒绝,反射调用[PermissionRequestFailed]注解的方法,支持无参或只有一个[PermissionDetail]参数
     */
    private fun onDenied(joinPoint: ProceedingJoinPoint, permissionDetail: PermissionDetail) {
        val cls: Class<*> = joinPoint.`this`.javaClass
        val methods = cls.declaredMethods
        if (methods.isEmpty()) return
        methods.firstOrNull {
            it.isAnnotationPresent(PermissionRequestFailed::class.java)
        }?.apply {
            isAccessible = true
            val types = parameterTypes
            if (types.isEmpty()) {
                invoke(joinPoint.`this`)
            } else if (types.size == 1) {
                invoke(joinPoint.`this`, permissionDetail)
            }
        }
    }

    /**
     * Desc: 从切面上下文中获取当前Activity
     */
    private fun findActivityFromContext(any: Any): FragmentActivity? {
        if (any is Fragment) {
            val activity = any.activity
            if (activity != null && !activity.isDestroyed) {
                return activity
            }
        }
        if (any is FragmentActivity) {
            return any
        }
        // 获取栈顶activity
        val curActivity = getTopActivity()
        if (curActivity is FragmentActivity) {
            return curActivity
        }
        return null
    }
}

利用Fragment进行权限申请

/**
 * Desc: 使用registerForActivityResult进行权限申请
 *
 * @param permissions 权限列表
 * @param requestCode 请求码
 *
 * @return 权限请求结果详情[PermissionDetail]
 */
internal suspend fun FragmentActivity.requestPermissionsForResult(permissions: Array<String>, requestCode: Int): PermissionDetail {
    return suspendCoroutine { continuation ->
        val fragment = Fragment()
        var launch: ActivityResultLauncher<Array<String>>? = null
        // 由于registerForActivityResult必须在fragment或者activity onCreate之前注册,所以这里每次都添加一个空fragment进行注册
        launch = fragment.registerForActivityResult(ActivityResultContracts.RequestMultiplePermissions()) {
            if (it.isEmpty()) {
                launch?.unregister()
                continuation.resumeWithException(IllegalArgumentException("权限${permissions.contentToString()}申请结果为空。注意:权限请求只有第一次有效,请检查是否有重复调用权限请求的地方"))
                return@registerForActivityResult
            }
            // 授予权限列表 
            val grantedPermissions = mutableListOf<String>()
            // 拒绝权限列表
            val deniedPermissions = mutableListOf<String>()
            // 是否拒绝且勾选了不再提示
            var rejectRemind = false
            it.forEach { entry ->
                val permission = entry.key
                if (!entry.value) {
                    deniedPermissions.add(permission)
                    if (!rejectRemind) {
                        rejectRemind = !ActivityCompat.shouldShowRequestPermissionRationale(this, permission)
                    }
                } else {
                    grantedPermissions.add(permission)
                }
            }
            // 权限请求结果详情
            val detail = PermissionDetail(
                requestCode = requestCode,
                grantedPermissions = grantedPermissions,
                deniedPermissions = deniedPermissions,
                rejectRemind = rejectRemind,
            )
            supportFragmentManager.beginTransaction().remove(fragment).commitAllowingStateLoss()
            continuation.resume(detail)
        }
        supportFragmentManager.beginTransaction().add(fragment, "PermissionRequestFragment").commitAllowingStateLoss()
        // 等待fragment创建完成 
        fragment.lifecycleScope.launchWhenResumed {
            launch.launch(permissions)
        }
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,386评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,939评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,851评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,953评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,971评论 5 369
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,784评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,126评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,765评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,148评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,744评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,858评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,479评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,080评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,053评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,278评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,245评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,590评论 2 343

推荐阅读更多精彩内容