如何使用通知图标修复小米特定的RemoteServiceException?

时间:2018-12-10 17:23:29

标签: android push-notification xiaomi

我们在Android 6和7上有很多针对小米手机的崩溃:

Fatal Exception: android.app.RemoteServiceException: Bad notification posted from package x.y.z: Couldn't create icon: StatusBarIcon(icon=Icon(typ=RESOURCE pkg=x.y.z id=0x7f0200ad) visible user=0 )
   at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1715)
   at android.os.Handler.dispatchMessage(Handler.java:102)
   at android.os.Looper.loop(Looper.java:163)
   at android.app.ActivityThread.main(ActivityThread.java:6358)
   at java.lang.reflect.Method.invoke(Method.java)
   at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:880)
   at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:770)

我在网上发现了很多类似的崩溃报告和文章。这里是一些:

How to fix: android.app.RemoteServiceException: Bad notification posted from package *: Couldn't create icon: StatusBarIcon

https://medium.com/@Miqubel/the-story-of-a-hard-to-fix-bug-ac6ed819cb49

但是区别是我们仅在小米手机(Android 6和7)上有这些问题,而更新期间可能没有,因为同一用户在同一发行版中多次崩溃。

有趣的是,在这种情况下,我在网上找不到任何东西,我们周围也没有小米手机。

我将通知设置如下:

NotificationManager notificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
    if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) {
        int importance = NotificationManager.IMPORTANCE_HIGH;
        NotificationChannel notificationChannel = new NotificationChannel(NOTIFICATION_CHANNEL_ID, NOTIFICATION_CHANNEL_NAME, importance);
        notificationChannel.enableLights(true);
        notificationManager.createNotificationChannel(notificationChannel);
    }
    NotificationCompat.Builder mBuilder = new NotificationCompat.Builder(context, NOTIFICATION_CHANNEL_ID)
            .setSmallIcon(R.drawable.ic_notification)
            .setPriority(NotificationCompat.PRIORITY_MAX)
            .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher))
            .setContentText(body == null ? "" : body)
            .setAutoCancel(true)
            .setContentIntent(PendingIntent.getActivity(
                    context,
                    0,
                    pendingIntent,
                    PendingIntent.FLAG_UPDATE_CURRENT
            ));

我们也有Facebook通知,必须以类似的方式设置,但在不同的Notification类上。我不知道这是否相关。除了包装setSmallIcon和/或setLargeIcon方法到制造商和Android版本检查中以外,是否有人遇到过此问题或有任何建议来解决此问题?

编辑: 我找不到解决方案,但是这里有一些新想法:

  • 我们发布了一个新版本,但不包括小米用户 通知没有帮助!现在我认为问题是由 ActivityThread.java中的自定义代码。 MIUI可能会触发 某些事件从这里发出的通知。有几十个事件 在此处存储Android,但没有一个会触发通知。但 我们的图标出了点问题,因此它们崩溃了。

  • 但是我们的图标有什么问题?我们有一个ic_notification, 可能不用于此目的。另一方面,ic_launcher是 mipmap。也许是这个吗?但我找不到任何问题 关于小米和mipmaps。

  • 崩溃报告始终在多个应用版本中提到相同的资源ID:0x7f0200ad。由于某种原因,这特别吗?如何对我们的应用程序进行反向工程以获取其资源名称?

编辑2:

  • 我使用apktool对应用程序进行了反向工程,但资源ID不在public.xml中,这似乎与R.java等效。我们的ic_notification和ic_launcher在列表中具有不同的ID。那么,这是MIUI无法找到的系统资源吗?

编辑3:

  • 其他人也有相同问题的第一个证据

https://xiaomi.eu/community/threads/miui-9.47247/

  • 在波兰论坛上找到的临时解决方案:

https://pl.forum.elvenar.com/index.php?threads/problem-z-uruchomieniem-23566.3348/

最后一条评论翻译为: “我们暂时解决了小米的问题,请尝试在电话设置中禁用从Elvenar应用程序强制执行的通知。重新启动该应用程序后,该错误应会消失。”

编辑4:

我们正在使用ShortcutBadger(版本1.1.13)。这里说我们应该对小米徽章使用不同的方法:

https://github.com/leolin310148/ShortcutBadger/wiki/Xiaomi-Device-Support

在1.1.13版本之后,他们删除了对小米的默认支持,因此您必须使用上述链接中的通知。

受影响的其他人都使用吗?

2 个答案:

答案 0 :(得分:3)

我和用户有同样的问题。我相信这是由以下代码引起的-从我没有源代码的APK中反编译,来自shortcutbadger

ResolveInfo.getIconResource()返回了无效的资源ID(0x7f0200ad,在所有应用程序中相同,并且看起来仅在MIUI10上),因此崩溃。

iget-object v1, p0, Lme/leolin/shortcutbadger/impl/XiaomiHomeBadger;->a:Landroid/content/pm/ResolveInfo;

invoke-virtual/range {v1 .. v1}, Landroid/content/pm/ResolveInfo;->getIconResource()I

move-result v1

invoke-virtual {p1, v1}, Landroid/app/Notification$Builder;->setSmallIcon(I)Landroid/app/Notification$Builder;

作为用户,一个简单的解决方法是完全禁用该应用程序的通知-至少使其可用。

作为开发人员,只使用自己的图标或在快捷方式标记中添加一些错误处理可能会更容易。

builder.setSmallIcon(R.drawable.myicon);

更新

我知道现在发生了什么……shortcutbadge中有些奇怪的代码。 resolveInfo是默认的家庭启动器(MIUI home)活动,resolveInfo.getIconResource()是miui家庭的图标

@TargetApi(Build.VERSION_CODES.JELLY_BEAN)
private void tryNewMiuiBadge(Context context, int badgeCount) throws ShortcutBadgeException {
if (resolveInfo == null) {
    Intent intent = new Intent(Intent.ACTION_MAIN);
    intent.addCategory(Intent.CATEGORY_HOME);
    resolveInfo = context.getPackageManager().resolveActivity(intent, PackageManager.MATCH_DEFAULT_ONLY);
}

if (resolveInfo != null) {
    NotificationManager mNotificationManager = (NotificationManager) context
            .getSystemService(Context.NOTIFICATION_SERVICE);
    Notification.Builder builder = new Notification.Builder(context)
            .setContentTitle("")
            .setContentText("")
            .setSmallIcon(resolveInfo.getIconResource());
    Notification notification = builder.build();

并从miuihome.apk中反编译,这是0x7f0200ad。

<public type="drawable" name="icon_launcher" id="0x7f0200ad" />

那么,为什么第三方应用程序会尝试使用miui home的图标设置通知图标?是出于兼容性原因还是出于错误?我用上述代码段编写了一个简单的应用程序,在模拟器上进行了测试,但失败了,但没有崩溃该应用程序,在旧的MIUI上可能是相同的情况,因为setSmallIcon(resID)从自己的包中查找带有resID的图标。好消息是,这不是MIUI10的错误,它只应在使用上述代码的应用程序上发生。

答案 1 :(得分:2)

我们购买了Redmi Note 4X。这是发生了什么:

  • 设备使用的是MIUI 8.5。通知按预期方式工作。没有崩溃。

  • 我们通过OTA更新将其更新为MIUI 9.5。当我们打开应用程序或在某些屏幕之间切换时,奇怪的通知开始出现。仍然没有崩溃。

  • 我们通过OTA更新将其更新为MIUI 10.1。应用在使用先前版本的应用启动时崩溃。当我将ShortcutBadger更新为1.1.22时,崩溃消失了。显然,发生这种情况是因为小米设备没有在新版本的库中执行applyCount()方法(它们必须使用applyNotification()代替),但是我没有更深入地研究。

我向任何可以详细解释发生的事情的人开放问题和悬赏。否则,到目前为止我们对这个结果感到满意。对于下一个版本,我可能会尝试修复小米的徽章和通知,因为它们无法正确显示。