Android 8.0系统的通知栏适配
既然选择了远方,便只顾风雨兼程.
该图片来自网络资源,若有侵权,请留言,我将自行删除最近项目不是很忙,所以抽空了解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