Android 8.0+ 通知栏适配

作者: 049032247b6b | 来源:发表于2019-07-16 19:45 被阅读32次

    对于通知栏,大家都不陌生,应该算是设备的基础功能组件了吧,像推送消息,任务提醒,闹钟提示等等都需要借助设备的通知栏展现给用户,所以说一个醒目且友好的通知是很重要的,但是在这个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之前,如果想将消息显示在通知栏上面,基本上是先创建一个 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版本相关的封装,这里引用官网的一段文档看下解释

    各版本的通知栏创建方式已经介绍完了,另外,自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+ 通知栏适配

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