昨天Google在Google I / O上展示了基于新Firebase的新通知系统。我在Github上尝试了这个新的FCM(Firebase云消息传递)示例。
通知的图标始终是 ic_launcher ,尽管我已声明了特定的可绘制
为什么? 以下是处理消息的官方代码
public class AppFirebaseMessagingService extends FirebaseMessagingService {
/**
* Called when message is received.
*
* @param remoteMessage Object representing the message received from Firebase Cloud Messaging.
*/
// [START receive_message]
@Override
public void onMessageReceived(RemoteMessage remoteMessage) {
// If the application is in the foreground handle both data and notification messages here.
// Also if you intend on generating your own notifications as a result of a received FCM
// message, here is where that should be initiated. See sendNotification method below.
sendNotification(remoteMessage);
}
// [END receive_message]
/**
* Create and show a simple notification containing the received FCM message.
*
* @param remoteMessage FCM RemoteMessage received.
*/
private void sendNotification(RemoteMessage remoteMessage) {
Intent intent = new Intent(this, MainActivity.class);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0 /* Request code */, intent,
PendingIntent.FLAG_ONE_SHOT);
Uri defaultSoundUri= RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION);
// this is a my insertion looking for a solution
int icon = Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP ? R.drawable.myicon: R.mipmap.myicon;
NotificationCompat.Builder notificationBuilder = new NotificationCompat.Builder(this)
.setSmallIcon(icon)
.setContentTitle(remoteMessage.getFrom())
.setContentText(remoteMessage.getNotification().getBody())
.setAutoCancel(true)
.setSound(defaultSoundUri)
.setContentIntent(pendingIntent);
NotificationManager notificationManager =
(NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
notificationManager.notify(0 /* ID of notification */, notificationBuilder.build());
}
}
答案 0 :(得分:219)
不幸的是,这是SDK 9.0.0-9.6.1中Firebase通知的限制。当应用程序在后台时,启动器图标将从清单中使用(具有必要的Android着色),用于从控制台发送的消息。
但是,使用SDK 9.8.0,您可以覆盖默认值!在AndroidManifest.xml中,您可以设置以下字段来自定义图标和颜色:
<meta-data
android:name="com.google.firebase.messaging.default_notification_icon"
android:resource="@drawable/notification_icon" />
<meta-data android:name="com.google.firebase.messaging.default_notification_color"
android:resource="@color/google_blue" />
请注意,如果应用程序位于前台(或发送数据消息),您可以完全使用自己的逻辑来自定义显示。如果从HTTP / XMPP API发送消息,您也可以始终自定义图标。
答案 1 :(得分:31)
使用服务器实施向您的客户发送消息并使用 数据 类型的消息,而不是 通知 消息类型。
这将有助于您回复onMessageReceived
,无论您的应用是在后台还是前台,您都可以生成自定义通知
答案 2 :(得分:7)
atm他们正在处理该问题https://github.com/firebase/quickstart-android/issues/4
当您从Firebase控制台发送通知时,默认使用您的应用图标,Android系统会在通知栏中将该图标变为纯白色。
如果您对该结果不满意,您应该实施FirebaseMessagingService并在收到消息时手动创建通知。我们正在努力改善这一点,但目前这是唯一的方法。
编辑:使用SDK 9.8.0添加到 AndroidManifest.xml
<meta-data android:name="com.google.firebase.messaging.default_notification_icon" android:resource="@drawable/my_favorite_pic"/>
答案 3 :(得分:6)
我的解决方案类似于ATom的解决方案,但更容易实现。您不需要创建一个完全隐藏FirebaseMessagingService的类,您可以只覆盖接收Intent的方法(这是公共的,至少在版本9.6.1中),并从附加内容中显示信息。 “hacky”部分是方法名称确实是混淆的,并且每次将Firebase sdk更新为新版本时都会更改,但您可以通过使用Android Studio检查FirebaseMessagingService并查找需要的公共方法来快速查找一个Intent作为唯一的参数。在版本9.6.1中,它被称为zzm。 这是我的服务的样子:
public class MyNotificationService extends FirebaseMessagingService {
public void onMessageReceived(RemoteMessage remoteMessage) {
// do nothing
}
@Override
public void zzm(Intent intent) {
Intent launchIntent = new Intent(this, SplashScreenActivity.class);
launchIntent.setAction(Intent.ACTION_MAIN);
launchIntent.addCategory(Intent.CATEGORY_LAUNCHER);
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0 /* R equest code */, launchIntent,
PendingIntent.FLAG_ONE_SHOT);
Bitmap rawBitmap = BitmapFactory.decodeResource(getResources(),
R.mipmap.ic_launcher);
NotificationCompat.Builder notificationBuilder = new NotificationCompat.Builder(this)
.setSmallIcon(R.drawable.ic_notification)
.setLargeIcon(rawBitmap)
.setContentTitle(intent.getStringExtra("gcm.notification.title"))
.setContentText(intent.getStringExtra("gcm.notification.body"))
.setAutoCancel(true)
.setContentIntent(pendingIntent);
NotificationManager notificationManager =
(NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
notificationManager.notify(0 /* ID of notification */, notificationBuilder.build());
}
}
答案 4 :(得分:3)
只需将targetSdkVersion设置为19.通知图标将为彩色。 然后等待Firebase解决此问题。
答案 5 :(得分:3)
还有一种丑陋却又有效的方法。反编译FirebaseMessagingService.class并修改它的行为。然后将该类放在yout app中的正确包中,dex使用它而不是消息库本身中的类。它非常简单而且有效。
有方法:
private void zzo(Intent intent) {
Bundle bundle = intent.getExtras();
bundle.remove("android.support.content.wakelockid");
if (zza.zzac(bundle)) { // true if msg is notification sent from FirebaseConsole
if (!zza.zzdc((Context)this)) { // true if app is on foreground
zza.zzer((Context)this).zzas(bundle); // create notification
return;
}
// parse notification data to allow use it in onMessageReceived whe app is on foreground
if (FirebaseMessagingService.zzav(bundle)) {
zzb.zzo((Context)this, intent);
}
}
this.onMessageReceived(new RemoteMessage(bundle));
}
此代码来自版本9.4.0,由于混淆,方法将在不同版本中具有不同的名称。
答案 6 :(得分:2)
我正在通过FCM控制台和HTTP / JSON触发我的通知......结果相同。
我可以处理标题,完整信息,但图标始终是默认的白色圆圈:
代替代码中的自定义图标(setSmallIcon或setSmallIcon)或应用中的默认图标:
Intent intent = new Intent(this, MainActivity.class);
// use System.currentTimeMillis() to have a unique ID for the pending intent
PendingIntent pIntent = PendingIntent.getActivity(this, (int) System.currentTimeMillis(), intent, 0);
if (Build.VERSION.SDK_INT < 16) {
Notification n = new Notification.Builder(this)
.setContentTitle(messageTitle)
.setContentText(messageBody)
.setSmallIcon(R.mipmap.ic_launcher)
.setContentIntent(pIntent)
.setAutoCancel(true).getNotification();
NotificationManager notificationManager =
(NotificationManager) getSystemService(NOTIFICATION_SERVICE);
//notificationManager.notify(0, n);
notificationManager.notify(id, n);
} else {
Bitmap bm = BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher);
Notification n = new Notification.Builder(this)
.setContentTitle(messageTitle)
.setContentText(messageBody)
.setSmallIcon(R.drawable.ic_stat_ic_notification)
.setLargeIcon(bm)
.setContentIntent(pIntent)
.setAutoCancel(true).build();
NotificationManager notificationManager =
(NotificationManager) getSystemService(NOTIFICATION_SERVICE);
//notificationManager.notify(0, n);
notificationManager.notify(id, n);
}
答案 7 :(得分:2)
答案 8 :(得分:1)
写这个
<meta-data
android:name="com.google.firebase.messaging.default_notification_icon"
android:resource="@drawable/ic_notification" />
右移<application.....>
答案 9 :(得分:0)
我想为这个问题添加一个答案,因为我的问题很简单但很难注意到。特别是在创建com.google.firebase.messaging.default_notification_icon
时,我复制/粘贴了一个现有的元数据元素,该元素使用android:value
标签来指定其值。这不适用于通知图标,一旦将其更改为android:resource
,一切都会按预期进行。