我在使用 FCM 推送通知消息服务时遇到问题,因为我已覆盖了在应用程序处于前台时接收通知的handleIntent()
方法。我也在使用onMessageReceived()
方法。
但是当应用程序在后台运行时,我会收到 2 条通知,其中一条仅打开应用程序并运行MainActivity
,而另一条则以我想要的方式打开应用程序。
仅供参考:当我在前台时,我收到的通知正是我希望它打开的方式。
这是我在下面编写的代码:
public class MyFirebaseMessagingService extends FirebaseMessagingService {
private final String NOTIFICATION_TYPE = "type";
private final String NOTIFICATION_ID = "id";
private final String NOTIFICATION_TYPE_PRODUCT_DETAIL = "productdetail";
@Override
public void onMessageReceived(RemoteMessage remoteMessage) {
super.onMessageReceived(remoteMessage);
String title = remoteMessage.getNotification().getTitle();R
String body = remoteMessage.getNotification().getBody();
String token = remoteMessage.getFrom();
Log.d("FireBase TAG: ", token);
// Check if message contains a data payload.
if (remoteMessage.getData().size() > 0) {
Log.d("FireBaseMessageService","FireBase Data payload : " + remoteMessage.getData());
}
// Check if message contains a notification payload.
if (remoteMessage.getNotification() != null) {
Log.d(TAG, "Message Notification Body: " + remoteMessage.getNotification().getBody());
}
}
@Override
public void handleIntent(Intent intent) {
super.handleIntent(intent);
String type = intent.getExtras().getString(NOTIFICATION_TYPE, "");
int id = 0;
try {
id = Integer.valueOf(intent.getExtras().getString(NOTIFICATION_ID, ""));
} catch (NumberFormatException e) {
e.printStackTrace();
}
//Intents
Intent mainIntent = MainActivity.newIntent(this);
Intent editProfileIntent = EditProfileActivity.newIntent(this);
Intent settingsIntent = SettingsActivity.newIntent(this);
Intent productIntent = ProductActivity.newNotificationIntent(this, id, false, true);
if (UserManager.getSingleton().isUserLoggedIn(getApplicationContext())) {
PendingIntent pendingIntent;
if (type.equalsIgnoreCase(NOTIFICATION_TYPE_PRODUCT_DETAIL)) {
TaskStackBuilder stackBuilder = TaskStackBuilder.create(this);
stackBuilder.addParentStack(MainActivity.class);
stackBuilder.addNextIntent(mainIntent);
stackBuilder.addNextIntent(productIntent);
editProfileIntent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
pendingIntent = stackBuilder.getPendingIntent(0, PendingIntent.FLAG_ONE_SHOT);
}
else {
pendingIntent = PendingIntent.getActivity(this, 0, productIntent,
PendingIntent.FLAG_ONE_SHOT);
}
NotificationCompat.Builder builder = new NotificationCompat.Builder(this)
.setSmallIcon(R.mipmap.ic_launcher)
.setContentTitle(intent.getExtras().getString("gcm.notification.title"))
.setContentText(intent.getExtras().getString("gcm.notification.body"))
.setDefaults(NotificationCompat.DEFAULT_VIBRATE)
.setPriority(NotificationCompat.PRIORITY_MAX)
.setContentIntent(pendingIntent);
NotificationManager manager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
manager.notify(0, builder.build());
}
}
}
我已经从onMessageReceived()
方法中删除了NotificationCompat.Builder
。但我仍然在后台收到两个通知。
应用格拉德尔 :
compile 'com.google.firebase:firebase-core:11.4.2' //Firebase
compile 'com.google.firebase:firebase-messaging:11.4.2' //Firebase Cloud Messaging
我尝试在线搜索解决方案,但不幸的是没有指向Android的解决方案。
您正在将通知内容处理到handleIntent(Intent intent)
中。您可能应该移除super.handleIntent(intent);
,以防止 Firebase 系统在应用处于后台时处理通知。
解决方案:删除super.handleIntent(intent);
只需创建一个 sendnotification(( 方法并设置您想要传递的任何参数,例如 body,即 sendnotification(字符串正文(。使用挂起的意图来启动您的活动,当您单击通知时,您的应用程序将数据解析到清单中定义的启动器活动,因此一旦启动器活动中有数据,您就可以使用 意图将数据发送到其他活动。
我认为 .setContentText("( 被调用了 1 次以上,您是否收到两次相同的通知?
完美运行的通知由您的代码生成,但是当您的应用程序不在前台时,android系统将为您生成通知。在这种情况下,如果您无法控制发送意图中的数据,则打开所需的活动。在这种情况下,您必须对服务器有效负载进行一些修改。您必须在有效负载中添加click_action,这就是 android 系统识别目标活动的方式。
有效负载示例:
{ "通知": {
"title":"Notification title",
"body":"Notification body",
"click_action":"<Your_destination_activity>",
}, "data":{
"param1":"value1",
"param2":"value2"
},
"priority":"high",
}
参考:https://firebase.google.com/docs/cloud-messaging/http-server-ref
是的,当您在后台应用时,您将在系统托盘上收到推送,因此系统托盘将创建带有通知标题和消息的推送。当您单击推送时,您的初始启动器活动(在清单中称为启动器(将打开。
您可以在启动器活动(捆绑包(中获取通知数据。
private void handlePush() {
Intent intent = null;
if (bundle.getString("push_type") != null && bundle.getString("push_type").length() > 0) {
switch (bundle.getString("push_type")) {
case PUSH_TYPE_FOLLOW_USER: {
intent = new Intent(this, ProfileExternalActivity.class);
intent.putExtra(Constants.USER_ID, Integer.parseInt(bundle.getString("id")));
intent.putExtra(Constants.FROM_PUSH_NOTIFICATION_SPLASH, true);
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK | Intent.FLAG_ACTIVITY_NEW_TASK);
break;
}
}
if (intent != null)
startActivity(intent);
finish();
}
}
并且您需要检查活动是否有数据
if (bundle != null)
handlePush();
else //your next activity
仅供参考:https://firebase.google.com/docs/cloud-messaging/android/receive
或
你可以在通知中获取有效载荷对象而不是数据对象,如果你的通知对象中有有效载荷对象,请在你的onMessageReceived((上推送所有收到的时间。
对于仍然有此问题的人:
这是防止此行为的黑客。我已经看了一遍,似乎关于这个的信息很少,但是如果您将发送的实际消息保存在共享首选项中,然后在onRecieve中对该值进行检查,则可以轻松防止这种情况。唯一的缺点是您的用户无法以通知的形式连续发送两次完全相同的消息(但这无论如何都会很烦人(。例:
@Override
public void onMessageReceived(RemoteMessage remoteMessage) {
SharedPreferences Settings = getDefaultSharedPreferences(context);
SharedPreferences.Editor editor = Settings.edit();
message = remoteMessage.getNotification().getBody();
from = remoteMessage.getFrom(); //this comes through as the topic, oddly...
if(from.equals("/topics/"+userID+deviceID+"all")) {
if(!message.equals(Settings.getString("messageall",null))) {//this filters any dupe messages
utils.postNotification(title, message, context, extra, "messages");//create notification
editor.putString("messageall", message);//always update to the last message
editor.commit();
}
}
}