如何通过Kotlin协程, 简化"连续依次弹窗(Dialog队列)"的需求

7,430 阅读1分钟

效果预览

r2t33-r5h2v.gif

代码预览

lifecycleScope.launch {
    showDialog("签到活动", "签到领10000币") // 直到dialog被关闭, 才会继续运行下一行
    showDialog("新手任务", "做任务领20000币") // 直到dialog被关闭, 才会继续运行下一行
    showDialog("首充奖励", "首充6元送神装")
}

代码实现

要做到上一个showDialig()在关闭时才继续运行下一个函数,需要用到协程挂起的特性, 然后在 OnDismiss()回调中将协程恢复, 为了将这种基于回调的方法包装成协程挂起函数, 可以使用suspendCancellableCoroutine函数

suspend fun showDialog(title: String, content: String) = suspendCancellableCoroutine { continuation ->
    MaterialAlertDialogBuilder(this)
        .setTitle(title)
        .setMessage(content)
        .setPositiveButton("我知道了") { dialog, which ->
            dialog.dismiss()
        }
        .setOnDismissListener {
            continuation.resume(Unit)
        }
        .show()
}

进阶玩法

跳过某些弹窗, 例如第二个弹窗只显示一次

suspend fun showDialogOnce(title: String, content: String) = suspendCancellableCoroutine { continuation ->
    val showed = SPUtils.getInstance().getBoolean(title) // SharedPreferences工具类
    if (showed) {
        continuation.resume(Unit)
        return@suspendCancellableCoroutine
    }
    MaterialAlertDialogBuilder(this)
        .setTitle(title)
        .setMessage(content)
        .setPositiveButton("我知道了") { dialog, which ->
            dialog.dismiss()
        }
        .setOnDismissListener {
            continuation.resume(Unit)
        }
        .show()
    SPUtils.getInstance().put(title, true)
}

调用时只需要这样:

lifecycleScope.launch {
    showDialog("签到活动", "签到领10000币")
    showDialogOnce("新手任务", "做任务领20000币")
    showDialog("首充奖励", "首充6元送神装")
}

这样'新手任务'的弹窗只会首次弹出, 后续只会弹出第一和第三个