文章目录

  • 前言
  • 一、Android8.0 Notification(通知)
  • 1.创建NotificationManager(通知管理器)
  • 2.发送通知
  • 二、手机桌面角标效果
  • 1、华为手机
  • 2、小米手机
  • 后记(完整代码)



前言

android8.0之后 Notification添加了NotificationChannel(渠道)的特性,如果App的targetSDKVersion>=26,没有设置channel通知渠道的话,就会导致通知无法展示,而且新的版本中把振动、音效和灯效等相关效果放在了通知渠道中控制,这样用户就可以有选择的控制应用的某一类通知的通知效果,而不像之前版本中应用所有通知都受控于一种设置。
开发过程中需要在APP中加一个消息提醒,要在桌面的APP图标加一个数量气泡效果,百度了。。。才发现一个小小功能这么麻烦,不同品牌的手机需要不同的实现方式,这里实现了小米和华为手机的效果,其他品牌的手机没找到实现方式


一、Android8.0 Notification(通知)

这里只实现了

1.创建NotificationManager(通知管理器)

使用通知时需要先创建NotificationManager实例,android8.0之后创建NotificationManager时需要添加NotificationChannel。
代码如下:

if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O){//8.0之后的创建方式
            NotificationChannel channel = new NotificationChannel(channel_id,channel_name, NotificationManager.IMPORTANCE_DEFAULT);
            channel.setDescription(description);
            channel.canBypassDnd();  //是否绕过请勿打扰模式
            channel.setLockscreenVisibility(Notification.VISIBILITY_SECRET); //锁屏显示通知
            channel.canShowBadge(); //桌面launcher的消息角标
            channel.getAudioAttributes(); //获取系统通知响铃声音的配置
            channel.getGroup();  //获取通知取到组
            channel.setBypassDnd(true);//设置可绕过  请勿打扰模式
            notificationManager = (NotificationManager)   getSystemService(Context.NOTIFICATION_SERVICE);
            notificationManager.createNotificationChannel(channel);
        }else{//8.0之前版本的创建方式
            notificationManager = (NotificationManager)    getSystemService(Context.NOTIFICATION_SERVICE);
        }

2.发送通知

Notification(通知)实例需要使用NotificationCompat,使用NotificationManager来发送通知
代码如下:

Notification notification = new NotificationCompat.Builder(this,channel_id)
                .setContentTitle(title)
                .setContentText(content)
                .setWhen(System.currentTimeMillis())
                .setSmallIcon(R.mipmap.ic_launcher)
                .setLargeIcon(BitmapFactory.decodeResource(getResources(),R.mipmap.ic_launcher))
                .setPriority(NotificationCompat.PRIORITY_DEFAULT)
                .build();
        notificationManager.notify(0,notification);//发送通知

二、手机桌面角标效果

这里只实现了华为和小米手机的角标效果,其他品牌的手机没有找到方法。。。

1、华为手机

官方文档

代码如下:

Bundle bunlde = new Bundle();
        bunlde.putString("package", "xxx"); // APP包名
        bunlde.putString("class", "xxxx"); // 点击图标进入的页面
        bunlde.putInt("badgenumber", count);
        this.getContentResolver().call(Uri.parse("content://com.huawei.android.launcher.settings/badge/"), "change_badge", null, bunlde);

点击图标进入程序后,角标不会自动消失还需要再次设置清零

2、小米手机

官方文档

代码如下:

private Notification setXiaoMiMarkNum(int count, Notification notification) throws Exception {
        Field field = notification.getClass().getDeclaredField("extraNotification");
        Object extraNotification = field.get(notification);
        Method method = extraNotification.getClass().getDeclaredMethod("setMessageCount", int.class);
        method.invoke(extraNotification, count);
        return notification;
    }

小米手机必须发送Notification(通知)才能设置角标而且点击图标进入程序后会自动清楚角标效果,不需要再次设置清除

后记(完整代码)

public class NotificationUtils extends ContextWrapper {
    String channel_id="myChannelId";
    String channel_name="myChannelName";
    String description = "this is myChannel's description";
    private NotificationManager notificationManager;
    private Context context;

    public NotificationUtils(Context context) {
        super(context);
        createNotificationChannel();
        this.context = context;
    }
    
    private void createNotificationChannel(){
        //Android8.0(API26)以上需要调用下列方法,但低版本由于支持库旧,不支持调用
        if(Build.VERSION.SDK_INT >= Build.VERSION_CODES.O){
            NotificationChannel channel = new NotificationChannel(channel_id,channel_name, NotificationManager.IMPORTANCE_DEFAULT);
            channel.setDescription(description);
            channel.canBypassDnd();  //是否绕过请勿打扰模式
            channel.setLockscreenVisibility(Notification.VISIBILITY_SECRET); //锁屏显示通知
            channel.canShowBadge(); //桌面launcher的消息角标
            channel.getAudioAttributes(); //获取系统通知响铃声音的配置
            channel.getGroup();  //获取通知取到组
            channel.setBypassDnd(true);//设置可绕过  请勿打扰模式
            notificationManager = (NotificationManager)   getSystemService(Context.NOTIFICATION_SERVICE);
            notificationManager.createNotificationChannel(channel);
        }else{
            notificationManager = (NotificationManager)    getSystemService(Context.NOTIFICATION_SERVICE);
        }

    }

    /**
     * 发送通知带桌面角标
     */
    public void sendNotificationAndMark(String title, String content,int count) {
        Notification notification = new NotificationCompat.Builder(this,channel_id)
                .setContentTitle(title)
                .setContentText(content)
                .setWhen(System.currentTimeMillis())
                .setSmallIcon(R.mipmap.ic_launcher)
                .setLargeIcon(BitmapFactory.decodeResource(getResources(),R.mipmap.ic_launcher))
                .setPriority(NotificationCompat.PRIORITY_DEFAULT)
                .build();
        notificationManager.notify(10086,notification);

        try{

            String manufacturer = Build.MANUFACTURER;
            if (manufacturer.equalsIgnoreCase("HUAWEI")) {//华为手机
                setHuaWeiMarkNum(count);
                notificationManager.notify(10086,notification);
            } else if (manufacturer.equalsIgnoreCase("Xiaomi")) {//小米
                notificationManager.notify(10086,setXiaoMiMarkNum(count,notification));
            } else {
                notificationManager.notify(10086,notification);
            }
        }catch (Exception e){
            Log.e("发送通知","发送异常=="+e.toString());
        }
    }

    private void setHuaWeiMarkNum(int count) throws Exception {
        Bundle bunlde = new Bundle();
        bunlde.putString("package", "xxx"); // APP包名
        bunlde.putString("class", "xxxx"); // 点击图标进入的页面
        bunlde.putInt("badgenumber", count);
        this.getContentResolver().call(Uri.parse("content://com.huawei.android.launcher.settings/badge/"), "change_badge", null, bunlde);
    }

    private Notification setXiaoMiMarkNum(int count, Notification notification) throws Exception {
        Field field = notification.getClass().getDeclaredField("extraNotification");
        Object extraNotification = field.get(notification);
        Method method = extraNotification.getClass().getDeclaredMethod("setMessageCount", int.class);
        method.invoke(extraNotification, count);
        return notification;
    }
}