美文网首页android适配
极光推送配合桌面图标角标的使用

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

作者: 有想法的小刘 | 来源:发表于2017-10-11 01:10 被阅读924次

    做极光推送,发现大部分手机收到推送信息,桌面图标没有角标显示未读取数量,才发现 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 是多余的,这里偷懒一下

    <receiver
                android:name=".MyJpushReceiver"
                android:enabled="true">
                <intent-filter>
                    <action android:name="cn.jpush.android.intent.REGISTRATION" />
                    <action android:name="cn.jpush.android.intent.MESSAGE_RECEIVED" />
                    <action android:name="cn.jpush.android.intent.NOTIFICATION_RECEIVED" />
                    <action android:name="cn.jpush.android.intent.NOTIFICATION_OPENED" />
                    <action android:name="cn.jpush.android.intent.NOTIFICATION_CLICK_ACTION" />
                    <action android:name="cn.jpush.android.intent.CONNECTION" />
                    <action android:name="com.rg.fragmentdemo.REMOVE_BADGE"/>
    
                    <category android:name="com.rg.fragmentdemo" />
                </intent-filter>
            </receiver>
    

    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 :当然,这里只是我的一种解决思路,角标主要是粗略地监听存储通知数量,不建议读者这么使用,因为这么做不完善,比方说手机的一键清除通知、重启手机之类的不可控因素,没法做到准确控制数量。不过这方法倒是可用户类似邮件是否读取的情况。

    相关文章

      网友评论

        本文标题:极光推送配合桌面图标角标的使用

        本文链接:https://www.haomeiwen.com/subject/pdlbyxtx.html