缺少可变性标志:带有 NavDeepLinkBuilder 的 Android 12 待定意图

Posted

技术标签:

【中文标题】缺少可变性标志:带有 NavDeepLinkBuilder 的 Android 12 待定意图【英文标题】:Missing mutability flags: Android 12 pending intents with NavDeepLinkBuilder 【发布时间】:2021-12-18 01:54:00 【问题描述】:

我已经开始在 android 12 上测试我的应用是否存在问题,并且收到了一些关于为主屏幕小部件设置的待处理意图上的可变性标志的警告。这些可变性标志现在在 SDK 31 中是强制性的。在 31 之前,此代码可以工作,但会出现 lint 警告:

val myIntent = Intent(context, MainActivity::class.java)
val myPendingIntent = PendingIntent.getActivity(context, 0, myIntent, PendingIntent.FLAG_UPDATE_CURRENT )
myRemoteViews.setOnClickPendingIntent(R.id.widget_button, myPendingIntent)

在 Android 12 上,需要添加第二个标志,如下所示:

val myIntent = Intent(context, MainActivity::class.java)
val myPendingIntent = PendingIntent.getActivity(context, 0, myIntent, PendingIntent.FLAG_UPDATE_CURRENT or PendingIntent.FLAG_IMMUTABLE)
myRemoteViews.setOnClickPendingIntent(R.id.widget_button, myPendingIntent)

到目前为止一切顺利。但是,对于某些小部件操作,不仅需要导航到应用程序,还需要导航到特定片段并添加片段参数。对于这些情况,我一直在使用 NavDeepLinkBuilder 创建 pendingIntent https://developer.android.google.cn/reference/kotlin/androidx/navigation/NavDeepLinkBuilder

val myPendingIntent = NavDeepLinkBuilder(context)
        .setGraph(R.navigation.nav_main)
        .setDestination(R.id.myFragment)
        .setArguments(
            myFragment.bundle(myArgs)
        )
        .createPendingIntent()
myRemoteViews.setOnClickPendingIntent(R.id.widget_button, myPendingIntent)

在您将构建目标设置为 sdk 31 并在 Android 12(模拟器)上构建之前,这非常有效。在这种情况下,它会在使用此堆栈跟踪创建小部件时崩溃:

   2021-11-03 17:41:27.792 20309-20309/com.package.debug E/AndroidRuntime: FATAL EXCEPTION: main
    Process: com.package.debug, PID: 20309
    java.lang.RuntimeException: Unable to start receiver com.package.widgets.mywidget.ui.MyWidgetProvider: java.lang.IllegalArgumentException: com.package.debug: Targeting S+ (version 31 and above) requires that one of FLAG_IMMUTABLE or FLAG_MUTABLE be specified when creating a PendingIntent.
    Strongly consider using FLAG_IMMUTABLE, only use FLAG_MUTABLE if some functionality depends on the PendingIntent being mutable, e.g. if it needs to be used with inline replies or bubbles.
        at android.app.ActivityThread.handleReceiver(ActivityThread.java:4321)
        at android.app.ActivityThread.access$1600(ActivityThread.java:247)
        at android.app.ActivityThread$H.handleMessage(ActivityThread.java:2068)
        at android.os.Handler.dispatchMessage(Handler.java:106)
        at android.os.Looper.loopOnce(Looper.java:201)
        at android.os.Looper.loop(Looper.java:288)
        at android.app.ActivityThread.main(ActivityThread.java:7842)
        at java.lang.reflect.Method.invoke(Native Method)
        at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:548)
        at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1003)
     Caused by: java.lang.IllegalArgumentException: com.package.debug: Targeting S+ (version 31 and above) requires that one of FLAG_IMMUTABLE or FLAG_MUTABLE be specified when creating a PendingIntent.
    Strongly consider using FLAG_IMMUTABLE, only use FLAG_MUTABLE if some functionality depends on the PendingIntent being mutable, e.g. if it needs to be used with inline replies or bubbles.
        at android.app.PendingIntent.checkFlags(PendingIntent.java:375)
        at android.app.PendingIntent.getActivitiesAsUser(PendingIntent.java:593)
        at android.app.PendingIntent.getActivities(PendingIntent.java:575)
        at androidx.core.app.TaskStackBuilder.getPendingIntent(TaskStackBuilder.java:341)
        at androidx.core.app.TaskStackBuilder.getPendingIntent(TaskStackBuilder.java:312)
        at androidx.navigation.NavDeepLinkBuilder.createPendingIntent(NavDeepLinkBuilder.java:260)
        at com.ms1881.widgets.listwidget.ui.ListWidgetProvider.buildRemoteViews(ListWidgetProvider.kt:91)
        at com.ms1881.widgets.listwidget.ui.ListWidgetProvider.updateListWidget(ListWidgetProvider.kt:44)
        at com.ms1881.widgets.listwidget.ui.ListWidgetProvider.onUpdate(ListWidgetProvider.kt:39)
        at android.appwidget.AppWidgetProvider.onReceive(AppWidgetProvider.java:66)
        at com.ms1881.widgets.listwidget.ui.ListWidgetProvider.onReceive(ListWidgetProvider.kt:169)
        at android.app.ActivityThread.handleReceiver(ActivityThread.java:4312)
            ... 9 more

所以问题是:在使用 NavDeepLinkBuilder 构建待处理意图时,如何设置强制可变性标志?我在 NavDeepLinkBuilder 的文档中找不到任何相关信息。可以像在第二个代码 sn-p 中那样设置挂起的意图,我可以使用意图操作或额外的来设置“地址”以到达所需的片段,但我失去了将捆绑参数发送到的可能性片段本身。感谢所有有关如何解决此问题的意见。

【问题讨论】:

【参考方案1】:

根据Navigation 2.4.0-alpha04 release notes:

NavDeepLinkBuilder 现在将PendingIntent.FLAG_IMMUTABLE 添加到createPendingIntent() 返回的PendingIntent,确保此 API 在针对 Android 12 时按预期工作。(If8c52)

如果您想要修复此问题,您需要升级到 Navigation 2.4.0-alpha04 或更高版本(当前为 2.4.0-beta01)。

【讨论】:

我使用的是 Navigation 2.3.5,没有考虑查看发行说明以了解有关此问题的更改。感谢您的输入(以及导航框架) 导航版本 2.4.0+ 包括多个 backstack。所以,我不能使用这个最新版本。如何在不更新到最新版本的情况下修复此崩溃,或者如何在最新版本中停止多个 backstack? 我也遇到了这个问题,2.4.0+版本的一些bug阻止我碰到依赖版本。

以上是关于缺少可变性标志:带有 NavDeepLinkBuilder 的 Android 12 待定意图的主要内容,如果未能解决你的问题,请参考以下文章

在 C 中使用 ioctl() 设置不可变标志

SSL Cookie 中缺少安全标志 (http-cookie-secure-flag)

在固定区域中具有可变内容和控制按钮的表单

spark UDF 怎么实现参数数量可变,不传参时候,有默认值。见下图?

交通标志识别基于matlab SIFT交通标志识别含Matlab源码 717期

交通标志识别基于SIFT特征实现交通标志识别matlab源码含GUI