做极光推送,发现大部分手机收到推送信息,桌面图标没有角标显示未读取数量,才发现 google 原生不提供方法显示角标,但是国内手机厂商逐渐开放添加角标的 api(仿 ios ) ,搜了下各种开源框架,最终锁定使用 ShortcutBadger。
- ShortcutBadger.applyCount(context, badgeCount) ,小米除外的手机调用该方法设置桌面图标的角标
- ShortcutBadger.applyNotification(context, mNotification, badgeCount) ,小米手机调用该方法设置桌面图标的角标
因此可以使用以下思路解决问题:
- badgeCount 默认为0,本地保存数值,接受到推送消息时,+1
- 点击通知跳转到指定的 Activity 时,消耗了该消息,因此 badgeCount - 1
- app 的入口 Activity 在 onCreate 和 onDestory 方法里,调用清除
badgeCount,即清除角标数据 - 因为小米手机设置角标的时,需要传 notification ,所以这里我们可以获取 notification 的 id、标题和内容,首先 cancel 掉推送的 notification ,然后再次添加相同 id、title、content,模拟推送消息
解决问题
AndroidManifest.xml ,添加接收推送消息的 receiver ,其实部分 action 是多余的,这里偷懒一下
MyJpushReceiver 里监听接收推送通知、点击通知、和清除角标的 action
public class MyJpushReceiver extends BroadcastReceiver {
private int notificationId = 0;//接受的notification的id
private int badgeCount = 0;//记录图标的角标数量
private static final String TAG = "MyJpushReceiver";
NotificationManager mManager;
Notification mNotification;
Notification.Builder mBuilder;
@Override
public void onReceive(Context context, Intent intent) {
Log.e(TAG, "onReceive: badgeCount=" + badgeCount);
Log.e(TAG, "action=" + intent.getAction());
try {
Bundle bundle = intent.getExtras();
if (JPushInterface.ACTION_NOTIFICATION_RECEIVED.equals(intent.getAction())) {
//接收到通知
badgeCount = PreferenceUtils.getInt(context, "badgeCount", 0);
badgeCount++;
PreferenceUtils.putInt(context, "badgeCount", badgeCount);
notificationId = bundle.getInt(JPushInterface.EXTRA_NOTIFICATION_ID);
Log.d(TAG, "[MyReceiver] 接收到推送下来的通知的ID: " + notificationId);//获取到通知调用
if (Build.MANUFACTURER.equalsIgnoreCase("xiaomi")) {
String title = context.getResources().getString(R.string.app_name);
String content = bundle.getString(JPushInterface.EXTRA_ALERT);
getNotification(context, title, content, notificationId);
ShortcutBadger.applyNotification(context, mNotification, badgeCount);
} else {
ShortcutBadger.applyCount(context, badgeCount);
}
} else if (JPushInterface.ACTION_NOTIFICATION_OPENED.equals(intent.getAction())) {
//接受到用户点击通知
Log.d(TAG, "[MyReceiver] 用户点击打开了通知");
badgeCount = PreferenceUtils.getInt(context, "badgeCount");
badgeCount--;
if (badgeCount < 0)
badgeCount = 0;
PreferenceUtils.putInt(context, "badgeCount", badgeCount);
notificationId = bundle.getInt(JPushInterface.EXTRA_NOTIFICATION_ID);
Log.d(TAG, "点击了id=" + notificationId + ",badgeCount=" + badgeCount);
if (Build.MANUFACTURER.equalsIgnoreCase("xiaomi")) {
Intent intent2 = new Intent();
intent2.setClass(context, com.rg.fragmentdemo.MainActivity.class);
intent2.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent2);
ShortcutBadger.applyNotification(context, mNotification, badgeCount);
} else {
ShortcutBadger.applyCount(context, badgeCount);
Intent intent3 = new Intent();
intent3.setClass(context, com.rg.fragmentdemo.MainActivity.class);
intent3.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent3);
}
} else if ("com.rg.fragmentdemo.REMOVE_BADGE".equals(intent.getAction())) {
//接收到移除所有badge
removeBadgeNum(context);
}
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 获取notification
*
* @param context
* @param title 标题
* @param message 内容
* @param notificationId notification的id
*/
private void getNotification(Context context, String title, String message, int notificationId) {
if (mManager == null) {
mManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);
}
mManager.cancel(notificationId);
mBuilder = new Notification.Builder(context);
mBuilder.setContentTitle(title);
mBuilder.setContentText(message);
mBuilder.setSmallIcon(R.mipmap.ic_launcher);
mNotification = mBuilder.build();
mNotification.flags = Notification.FLAG_AUTO_CANCEL;
//添加点击事件
Intent intent = new Intent();
intent.setAction(JPushInterface.ACTION_NOTIFICATION_OPENED);
intent.addCategory("com.rg.fragmentdemo");
Bundle bundle = new Bundle();
bundle.putInt(JPushInterface.EXTRA_NOTIFICATION_ID, notificationId);
intent.putExtras(bundle);
PendingIntent pendingIntent = PendingIntent.getBroadcast(context, notificationId, intent, PendingIntent.FLAG_UPDATE_CURRENT);
mBuilder.setContentIntent(pendingIntent);
mManager.notify(notificationId, mNotification);
}
private void removeBadgeNum(Context context) {
if (Build.MANUFACTURER.equalsIgnoreCase("xiaomi")) {
ShortcutBadger.applyNotification(context, mNotification, 0);
} else {
ShortcutBadger.applyCount(context, 0);
}
}
}
在 MainActivity 的 onCreate 和 onDestory 的方法里添加清除角标的方法
Intent intent = new Intent();
intent.setAction("com.rg.fragmentdemo.REMOVE_BADGE");
PreferenceUtils.putInt(this, "badgeCount", 0);
sendBroadcast(intent);
ps :当然,这里只是我的一种解决思路,角标主要是粗略地监听存储通知数量,不建议读者这么使用,因为这么做不完善,比方说手机的一键清除通知、重启手机之类的不可控因素,没法做到准确控制数量。不过这方法倒是可用户类似邮件是否读取的情况。