Android 8.0系统的通知栏适配

2019-03-04  本文已影响0人  coke613
既然选择了远方,便只顾风雨兼程.
该图片来自网络资源,若有侵权,请留言,我将自行删除

最近项目不是很忙,所以抽空了解RemoteViews控件. 咱们今天暂且不说RemoteViews,咱先说一说它的使用场景:通知栏以及桌面小部件.通知栏大家都不陌生,只要是通过NotificationManager.notify方法来实现的.但是你真的了解Notification,真的了解8.0之后Notification嘛.

首先说通知栏是Android原创的一个功能,虽说乔布斯一直在强调Android一直在抄袭iOS系统,但是通知栏的确是Android自创功能.在iOS 5.0系统之前,一直使用桌面badge,来提醒用户未读消息.在5.0之后,也加入了类似的通知功能.


iOS badge显示 该图片来自网络资源,若有侵权,请留言,我将自行删除
通知的基本用法

首先我们需要NotificationManager用来管理.可以通过Context.getSystemService拿到对应的NotificationManager.而getSystemService接收一个具体的系统服务名称,这里我们传入 NOTIFICATION_SERVICE.

 NotificationManager notificationManager = (NotificationManager) _mActivity.getSystemService(
                NOTIFICATION_SERVICE);

接下来需要一个Builder构造器来创建Notification对象.由于Android系统的每一个版本都会对通知这部分功能或多或少的都有些修改,Api 不稳定性问题在通知上就显得很严重!为了兼容这一方面的问题.我们可以使用support库中提供的兼容Api,
support-v4库中提供了NotificationCompat类,可以使用这个类的构造器来创建Notification对象.代码如下所示:

// 创建notification
 Notification notification = new NotificationCompat.Builder(_mActivity)
            .setContentTitle(" content title")
            .setContentText(" content")
            .setWhen(System.currentTimeMillis())
            .setSmallIcon(R.mipmap.ic_launcher)
            .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher))
            .setAutoCancel(true)   // 自动关闭状态栏
            .build();
 notificationManager.notify(1, notification);

其中setContentTitle()方法用于指定通知的标题内容.
setContentText()方法用于指定通知的内容.
setWhen()方法用于指定通知被创建的时间,这里参数类型是long,毫秒为单位.
setSmallIcon()方法用于显示通知的小图标.
setLargeIcon()方法用于显示通知的大图标.

代码写到这里,我们已经创建好了一个默认样式的通知,来吧,我们运行看看.
艾瑞巴得 莱斯够~~~

该图片来自网络资源,若有侵权,请留言,我将自行删除

然而 , 装X未遂.........

该图片来自网络资源,若有侵权,请留言,我将自行删除

程序为什么没响应呢?奥利奥 大家吃过吧,嗯 从Android O 开始,Google芭比引入了通知渠道概念.

通知渠道

通知渠道 顾明思议.每条通知都要属于一个对应的渠道.每一个App都可以自由的创建当前App拥有哪些通知渠道.但是这些通知渠道的控制权都是掌握在用户的手上的.用户可以自由的选择这些通知渠道的重要性,根据重要性来决定是否关闭这个渠道的通知.

来看一张微博,支付宝的通知渠道划分;


微博.png 支付宝.png

为什么要这样做呢?
提高了用户的自主性.有了通知渠道自主权之后,用户可以随意的自主选择更改通知.那些通知对于我来说是很重要的,我需要接收该通知.有那些通知对于我而言相当于骚扰通知,我可以针对性关闭.

创建通知渠道

如果你的项目中的targetSdkVersion 指定为26 或者更高,Android系统认为你的App已经做好了适配8.0系统的准备.


app.gradle

创建渠道

 if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
            String channelId = "chat";
            String channelName = "聊天消息";
            int importance = NotificationManager.IMPORTANCE_HIGH; // 高优先级
            createNotificationChannel(channelId, channelName, importance);
            channelId = "subscribe";
            channelName = "订阅消息";
            importance = NotificationManager.IMPORTANCE_DEFAULT;  
            createNotificationChannel(channelId, channelName, importance);
        }

因为通知渠道是8.0 新特性,在低于8.0版本中是没有通知渠道这个概念的.所以为了兼容低版本,我们在创建通知渠道的时候需要判断是否大于8.0 ,如果不做系统版本检查的话会在低版本的手机上崩溃.
上述代码我们模拟了两个场景,一个是聊天类型的通知,并将优先级设置为最高.还有一个场景是订阅类的,这类消息相比而言不是很重要,所以设置为IMPORTANCE_DEFAULT.

运行程序,并打开对应App的通知管理界面


image.png

我们刚刚创建的两个渠道现在已经展示出来了.用户可以点击进去,随意修改.比如取消震动,响铃,不在状态栏上显示.等等.

发送通知
 view.findViewById(R.id.bt_chat).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                // 创建通知管理器
                NotificationManager notificationManager = (NotificationManager) _mActivity.getSystemService(NOTIFICATION_SERVICE);
                Notification notification = new NotificationCompat.Builder(_mActivity, "chat")
                        .setContentTitle("您有一条新的消息")
                        .setContentText("中午没吃饭,现在肚子咕咕叫~~")
                        .setWhen(System.currentTimeMillis())
                        .setSmallIcon(R.mipmap.ic_launcher)
                        .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher))
                        .build();
                notificationManager.notify(1, notification);
            }
        });

view.findViewById(R.id.bt_subscribe).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                // 创建通知管理器
                NotificationManager notificationManager = (NotificationManager) _mActivity.getSystemService(NOTIFICATION_SERVICE);
                // 创建notification
                Notification notification = new NotificationCompat.Builder(_mActivity, "subscribe")
                        .setContentTitle("哇瑟 ! 您竟然中奖啦")
                        .setContentText("泉城寻找鲤锦,500万大奖等你拿!!!")
                        .setWhen(System.currentTimeMillis())
                        .setSmallIcon(R.mipmap.ic_launcher)
                        .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.mipmap.ic_launcher))
                        .build();
                notificationManager.notify(2, notification);
            }
        });

运行效果:


聊天消息 订阅消息

考虑一种情况:当用户动态的关闭通知了,也就意味着用户接收不到此类通知.那我们如何知道用户把某一通知渠道给关闭了呢?又如何通知用户打开手动打开?

渠道管理 代码如下:

 NotificationManager notificationManager = (NotificationManager) _mActivity.getSystemService(NOTIFICATION_SERVICE);
 NotificationChannel channel = notificationManager.getNotificationChannel("chat");
if (channel.getImportance() == NotificationManager.IMPORTANCE_NONE) {
         Intent intent = new Intent(Settings.ACTION_CHANNEL_NOTIFICATION_SETTINGS);
         intent.putExtra(Settings.EXTRA_APP_PACKAGE, _mActivity.getPackageName());
         intent.putExtra(Settings.EXTRA_CHANNEL_ID, channel.getId());
         startActivity(intent);
         Toast.makeText(_mActivity, "请手动将通知打开", Toast.LENGTH_SHORT).show();
}
image.gif

参照郭神码文:
https://mp.weixin.qq.com/s/Ez-G_9hzUCOjU8rRnsW8SA

上一篇下一篇

猜你喜欢

热点阅读