Android 8.0 Notification(通知)以及华为、小米手机软桌面角标的气泡效果


前言

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;
    }
}
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
来模拟这个场景: 首先,我们需要定义一个抽象的观察者接口,该接口包含一个更新方法,用于在主题状态发生变化时更新观察者状态。 ```java public interface Observer { void update(String message); } ``` 接下来,我们需要定义一个抽象的主题接口,该接口包含添加、删除和通知观察者的方法。 ```java public interface Subject { void registerObserver(Observer observer); void removeObserver(Observer observer); void notifyObservers(String message); } ``` 接下来,我们需要创建一个具体的主题类,该类实现了主题接口,并维护了一个观察者列表。在手机到货时,该类会遍历观察者列表并通知它们。 ```java import java.util.ArrayList; import java.util.List; public class HuaweiPhoneStore implements Subject { private List<Observer> observers; private String phoneModel; private boolean inStock; public HuaweiPhoneStore(String phoneModel) { observers = new ArrayList<>(); this.phoneModel = phoneModel; inStock = false; } public void setInStock(boolean inStock) { this.inStock = inStock; if (inStock) { notifyObservers(phoneModel + " is now available!"); } } @Override public void registerObserver(Observer observer) { observers.add(observer); } @Override public void removeObserver(Observer observer) { observers.remove(observer); } @Override public void notifyObservers(String message) { for (Observer observer : observers) { observer.update(message); } } } ``` 最后,我们需要创建具体的观察者类,该类实现了观察者接口,并在主题状态发生变化时收到通知。 ```java public class HuaweiPhoneFan implements Observer { private String name; public HuaweiPhoneFan(String name) { this.name = name; } @Override public void update(String message) { System.out.println(name + " received a notification: " + message); } } ``` 现在,我们可以创建一个 HuaweiPhoneStore 对象,并在其中注册几个 HuaweiPhoneFan 对象。当手机到货时,HuaweiPhoneStore 对象会通知所有注册的 HuaweiPhoneFan 对象。 ```java public class ObserverPatternDemo { public static void main(String[] args) { HuaweiPhoneStore store = new HuaweiPhoneStore("MateX"); HuaweiPhoneFan fan1 = new HuaweiPhoneFan("Alice"); HuaweiPhoneFan fan2 = new HuaweiPhoneFan("Bob"); HuaweiPhoneFan fan3 = new HuaweiPhoneFan("Charlie"); store.registerObserver(fan1); store.registerObserver(fan2); store.registerObserver(fan3); store.setInStock(true); } } ``` 输出如下: ``` Alice received a notification: MateX is now available! Bob received a notification: MateX is now available! Charlie received a notification: MateX is now available! ``` 这表明,当 HuaweiPhoneStore 对象通知观察者时,所有注册的 HuaweiPhoneFan 对象都收到了通知

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值