Android 8.0+(一) 通知栏适配

对于通知栏,大家都不陌生,应该算是设备的基础功能组件了吧,像推送消息,任务提醒,闹钟提示等等都需要借助设备的通知栏展现给用户,所以说一个醒目且友好的通知是很重要的,但是在这个android生态比较混乱的环境下,很多乱七八糟的通知,我明明不想接收,却见天儿的被各种app推送轰炸。

在Android 8.0 之前,有些做得好的厂商会针对每一个app提供一个通知的开关,但是这个权限被关了以后,就再也没法接收到通知了,以致于后面app的任何通知都无法展现给用户,除非用户再次开启app的通知开关。

也许Google意识到了这个问题,所以在Android 8.0上,NotificationChannel 应运而生。针对 targetSdkVersion在26以及上的设备,如果想要显示app通知,必须要注册对应通知的channel,一个app可以注册多个channel,而用户则可以自行设置某一个channel的开关状态,如果某一个channel的通知被关闭,那么该channel的任何通知都不会展现给用户,而其他channel的通知则不受影响

一图了解通知的基本组成

Android 8.0+(一) 通知栏适配_第1张图片

在Android 8.0之前,如果想将消息显示在通知栏上面,基本上是先创建一个 Notification ,然后由NotificationManager直接show() 是没有任何问题的

public void notify(int id) {
    Notification notification = new NotificationCompat.Builder(this)
                .setAutoCancel(true)
                .setSmallIcon(getSmallIcon())
                .setLargeIcon(getLargeIcon())
                .setContentTitle(title)
                .setContentText(body)
                .setContentIntent(pendingIntent)
                .build();

    getManager().notify(id, notification);
 }

但是如果在Android 8.0上执行这段代码是不会显示通知的,需要设置NotificationChannel才可以 ,而且设置channel之前,必须先注册channel

NotificationChannel channel = new NotificationChannel(
                “DEFAULT_CHANNEL_ID”,
                "推送通知", 
                NotificationManager.IMPORTANCE_HIGH
);
 getManager().createNotificationChannel(channel);

可以看到,NotificationChannel构造方法有三个参数,NotificationChannel(String id, CharSequence name,int importance),
id:channel的唯一标示,同一app的不同channel ,id不能重复
name:channel的描述信息
importance:该channel通知的重要程度

最后通过NotificationManager注册该channel, 注意: 必须在notify() 之前创建channel, 重复创建已有的channel不会有任何影响

现在,我们已经创建了一个channel,同样的,android 8.0及以上和7.0及以下 创建Notification的方式也有区别

 private NotificationCompat.Builder getNotificationBuilderByChannel(String channelId) {
        NotificationCompat.Builder builder;
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            builder = new NotificationCompat.Builder(getApplicationContext(), channelId);
        } else {
            builder = new NotificationCompat.Builder(this).setSound(soundUri);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {//8.0以下 && 7.0及以上 设置优先级
                builder.setPriority(NotificationManager.IMPORTANCE_HIGH);
            } else {
                builder.setPriority(NotificationCompat.PRIORITY_HIGH);
            }
        }
        return builder;
    }

由于8.0以后,NotificationCompat.Builder(Context context) 已经被废弃,所以创建方式还是采用最新的API比较好,虽然8.0以下并没有channel之说,但你如果看下源码,会发现其实NotificationCompat.Builder(Context context) 内部调用了NotificationCompat.Builder(Context context,String channelId) ,只不过channelId 传入的是个null而已 , 所以现在来看NotificationCompat.Builder的创建方式是一样的 。

   /** @deprecated */
        @Deprecated
        public Builder(Context context) {
            this(context, (String)null);
        }

另外需要注意的是,各版本设置通知的优先级的方式也有所不同,8.0及以上通知的优先级是在channel中设置的,8.0以下优先级是在NotificationCompat.Builder中设置的

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            NotificationChannel channel = new NotificationChannel(DEFAULT_CHANNEL,
                    "我的通知Channel", NotificationManager.IMPORTANCE_HIGH);         
    }

if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {//8.0以下 && 7.0及以上 设置优先级
             builder.setPriority(NotificationManager.IMPORTANCE_HIGH);
        } else {
             builder.setPriority(NotificationCompat.PRIORITY_HIGH);
    }

至于优先级中各个属性的具体介绍,请参考Android官方文档 android developers

可能你也注意到了,为什么我只在8.0以下设置了.setSound(soundUri); 那么 8.0+ 怎么设置通知的声音?
这是因为8.0+ 是在创建channel的时候设置通知音效的

AudioAttributes att = new AudioAttributes.Builder()
                    .setUsage(AudioAttributes.USAGE_NOTIFICATION)
                    .setContentType(AudioAttributes.CONTENT_TYPE_SPEECH)
                    .build();
NotificationChannel channel = new NotificationChannel(DEFAULT_CHANNEL,"我的通知 
                    Channel", NotificationManager.IMPORTANCE_HIGH);

channel.setSound(soundUri, att); 
getManager().createNotificationChannel(channel);

channel.setSound(soundUri, att); 除了设置音效的uri以外,还需要设置音频属性 ,这种方式有一种好处就是你可以给不同的channel设置不同的通知音效,如果没有设置自定义的uri,则将使用手机默认的通知铃声

现在,我们的通知栏已经适配了各个版本的api ,但是在某些手机上(oppo。。。),在应用安装完成以后,其通知默认是关闭状态,此时我们可以通过api判断通知开关是否开启,以此引导用户手动去设置中开启通知

NotificationManagerCompat.from(context).areNotificationsEnabled();

你应该也注意到了,我这里创建一个Notification使用的是NotificationCompat.Builder ,其实可以通过另一种方式 Notification.Builder 来创建 ,注意,这里选择前者的原因是NotificationCompat是兼容低版本的,而且内部帮我们做了一些api版本相关的封装,这里引用官网的一段文档看下解释

Android 8.0+(一) 通知栏适配_第2张图片

各版本的通知栏创建方式已经介绍完了,另外,自android8.0开始,支持在app启动图标上面显示通知红点,用户可以长按应用程序图标来查看该应用程序的通知。然后,用户可以通过左右滑动来关闭或处理来自该菜单的通知。下面简单介绍下如何在通知栏显示大文本和大图

通知栏显示文本默认只能显示一行,如果我们的文本太长了,将导致文本显示不全(以...结尾),此时我们可以在创建Notification时设置通知显示的样式,以此来实现大文本的显示

builder.setStyle(new NotificationCompat
                       .BigTextStyle()//大文本样式
                       .setBigContentTitle(title)//通知展开时显示的title
                       .bigText(body)//通知展开时显示的全部文本
);

Android通知栏支持显示大图 ,但是只能接收Bitmap类型的参数,所以如果你想要显示网络图片的话,需要自行将图片转换为Bitmap

builder.setStyle(new NotificationCompat
                       .BigPictureStyle()//大图模式
                       .setBigContentTitle(title)
                       .bigLargeIcon(imgBitmap)////通知展开时显示的缩略图
                       .bigPicture(imgBitmap)//通知展开时显示的大图
);

最后来看下实现的效果吧

你可能感兴趣的:(Android 8.0+(一) 通知栏适配)