极光推送配合桌面图标角标的使用

做极光推送,发现大部分手机收到推送信息,桌面图标没有角标显示未读取数量,才发现 google 原生不提供方法显示角标,但是国内手机厂商逐渐开放添加角标的 api(仿 ios ) ,搜了下各种开源框架,最终锁定使用 ShortcutBadger。

  • ShortcutBadger.applyCount(context, badgeCount) ,小米除外的手机调用该方法设置桌面图标的角标
  • ShortcutBadger.applyNotification(context, mNotification, badgeCount) ,小米手机调用该方法设置桌面图标的角标

因此可以使用以下思路解决问题:

  1. badgeCount 默认为0,本地保存数值,接受到推送消息时,+1
  2. 点击通知跳转到指定的 Activity 时,消耗了该消息,因此 badgeCount - 1
  3. app 的入口 Activity 在 onCreate 和 onDestory 方法里,调用清除
    badgeCount,即清除角标数据
  4. 因为小米手机设置角标的时,需要传 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 :当然,这里只是我的一种解决思路,角标主要是粗略地监听存储通知数量,不建议读者这么使用,因为这么做不完善,比方说手机的一键清除通知、重启手机之类的不可控因素,没法做到准确控制数量。不过这方法倒是可用户类似邮件是否读取的情况。

你可能感兴趣的:(极光推送配合桌面图标角标的使用)