前些天性致脖脖地点进入了developer.android.com,想看下通知这块内容,你懂的。首先映入眼帘的就是下面这玩意儿,翻译速度阔以哦!!!
通知渠道?啥玩意儿,啊,走过路过,千万不要错过啊,点进去瞧瞧呗。看到代码,必须先敲完运行下啊。什么情况,Api 26上根本就出不来什么通知啊。
心想谷歌还是严谨的,就选择相信了它。于是我打开了英文版。
通知渠道
先看什么是通知渠道。来看下官方解释:
Notification channels: Android 8.0 introduces notification channels that allow you to create a user-customizable channel for each type of notification you want to display.
啥意思?就是说这个面包我不想一个一个做了,做一个模子,把面粉放进去,就行了,形象吧。
没错,这些就是通知渠道。
我们可以看出,每个通知渠道都有一个名称,进去之后还有很多其他属性,如重要程序、通知圆点、闪烁灯等等。喂喂,那位拿8.0以下的手机试的同学,给我出去!
那有的同学要问了,怎样才能在App中显示一个通知渠道呢。问得好,我们直接上代码。
/**
* Oreo不用Priority了,用importance
* IMPORTANCE_NONE 关闭通知
* IMPORTANCE_MIN 开启通知,不会弹出,但没有提示音,状态栏中无显示
* IMPORTANCE_LOW 开启通知,不会弹出,不发出提示音,状态栏中显示
* IMPORTANCE_DEFAULT 开启通知,不会弹出,发出提示音,状态栏中显示
* IMPORTANCE_HIGH 开启通知,会弹出,发出提示音,状态栏中显示
*/
val channel = NotificationChannel("渠道ID",
"测试渠道名称",
NotificationManager.IMPORTANCE_HIGH)
// 获取NotificationManager
val notificationManager = getSystemService(Context.NOTIFICATION_SERVICE)
as NotificationManager
// 创建通知渠道
notificationManager.createNotificationChannel(channel)
就这么简单,对,就是这么简单暴力!
那如果想修改渠道属性怎么办呢?还是上面的代码,不过……
敲黑板,画重点,注意听了啊,只讲一遍,目前只有名称和描述可以直接改,其他的要看到修改后的结果只能先对App执行清除数据操作,那么问题来了,渠道还有哪些属性呢,我们不防点进去看看。
我们结合代码来看下。
// 设置提示音,IMPORTANCE_DEFAULT及以上才会有声音
channel.setSound(Uri.parse("..//aa.mp3"), AudioAttributes.Builder().build())
// 震动设置
channel.enableVibration(true)
// 设置震动模式,不设置使用系统默认
channel.vibrationPattern = longArrayOf(100, 200, 300, 400, 500, 400, 300, 200, 400)
// 闪烁指示灯设置
channel.enableLights(true)
// 指示灯颜色设置(不是每一个手机都支持哦)
channel.lightColor = Color.RED
// 屏幕锁定时通知显示方式(无法更改)
channel.lockscreenVisibility = Notification.VISIBILITY_SECRET
// 覆盖勿扰设置(无法更改)
channel.setBypassDnd(true)
// 获取NotificationManager
val notificationManager = getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager
// 通知描述
channel.description = "测试通知描述"
// 创建通知渠道
notificationManager.createNotificationChannel(channel)
其中“lockscreenVisibility”和“setBypassDnd”是无法生效的,因为从源码中来看,
只能被系统或排序服务(Android Notification Ranking Service)更改。
那么通知怎么跟通知渠道关联呢?Api26以上,创建通知需要提供渠道ID。
// 根据渠道创建通知
val notificationBuilder = Notification.Builder(this@TestActivity,
"渠道ID")
.setSmallIcon(R.drawable.ic_notification_logo)
.setContentTitle("测试通知标题")
.setContentText("测试通知内容")
// 弹出通知
notificationManager.notify(1, notificationBuilder.build())
接下来我们看看如果给渠道分组。渠道组对象是NotificationChannelGroup,也有一个ID和一个名称,创建一个渠道组很简单:
// 创建一个渠道组
val channelGroup = NotificationChannelGroup("测试组ID", "渠道组名")
// 绑定渠道组
channel.group = "测试组ID"
notificationManager.createNotificationChannelGroup(channelGroup)
上面代码创建渠道组和完成渠道与渠道组的绑定,如果渠道有绑定渠道组,必须先创建渠道组,再创建渠道哦。
关于渠道和渠道组的查询删除,就不用多说了吧。
// 查询所有当前用户的所有渠道
notificationManager.notificationChannelGroups
// 查询所有当前用户的所有渠道组
notificationManager.notificationChannels
// 根据ID删除渠道
notificationManager.deleteNotificationChannel("渠道ID")
// 根据ID删除渠道组
notificationManager.deleteNotificationChannel("测试组ID")
下面几个特性倒简单,我就稍微说说了。
通知角标
通知圆点可以在创建渠道时指定,要是创建时没指定修改时指定记得给App清除数据哦。
// 显示通知圆点
channel.setShowBadge(true)
打盹儿
自动消失
val notificationBuilder = Notification.Builder(this@TestActivity,
.setTimeoutAfter(5000L)
都很正常,然而坑往往就。。。
setSettingText
大概意思还是能猜到的,就是说8.0你可以通过INTENT_CATEGORY_NOTIFICATION_PREFERENCES
给通知加一个入口到你应用的通知设置里,而且还可以用setSettingsText()给入口指定文字。给个页面或demo会死啊,通知那一章节也只字不提这玩意儿了。再进setSettingText():
什么是affordance,难道不是下面这个齿轮嘛,难不成下面"MORE SETTINGS"这块文字能改?还是too young啊!
无奈只好谷歌呗。
我竟然寄希望去百度了。
看到第一条喜出望外啊!谁知道特么是官方文档的翻译,多的图就不截了,谁搜谁知道。可不能就这样放弃啊,今早从overflow上搜到了唯一一个,不过是自己昨晚提的。
试着从INTENT_CATEGORY_NOTIFICATION_PREFERENCES突破,总算有一点点进步。
本以为setSettingText() 可以改变上面文字,又一次失望了,昨晚看了youtube上的几个片子,均无介绍。就连兼容类NotificationCompat.Builder中都无setSettingText()方法。好吧,我决定看了源码之后再告诉你们,放我一条生路先!
再来看第二坑
傻乎乎地敲完了,试着移除几个通知,毛线都没有啊。
class NLService : NotificationListenerService(){
override fun onNotificationPosted(sbn: StatusBarNotification?) {
super.onNotificationPosted(sbn)
if(packageName == sbn?.packageName)
Log.i(TAG, "onNotificationPosted" + sbn.toString())
}
override fun onNotificationRemoved(sbn: StatusBarNotification?) {
super.onNotificationRemoved(sbn)
if(packageName == sbn?.packageName)
Log.i(TAG, "onNotificationRemoved" + sbn.toString())
}
原来是要手动开启通知访问权限啊。以Pixel为例,设置>应用和通知>高级>特殊应用权限->通知使用权。
还有坑?
看样子是能改背景色?好吧。
这是背景色吗?这是背景色吗?
机智如我,看到这下面这句话。
于是代码就成这样喽。
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
val notification = intent?.getParcelableExtra(ARG_NOTIFICATION)
if(notification != null){
startForeground(1, notification)
mHandler?.postDelayed(Runnable {
stopForeground(true)
}, TestNotificationColorService.DURATION_NOTIFICATION_DISMISS)
}
return super.onStartCommand(intent, flags, startId)
}
终于硬了一把。
最后一个终于不是坑了,大家放心踩,出了问题找我。
// If Showing message style notification, create some test data.
val showMessageStyleView = view?.findViewById(R.id.message_style_ck)
if(showMessageStyleView != null && showMessageStyleView.isChecked){
builder.setStyle(NotificationCompat.MessagingStyle(getString(R.string.test_display_name))
.setConversationTitle(getString(R.string.test_conversation_title))
.addMessage(getString(R.string.test_message_chat1), System.currentTimeMillis(), getString(R.string.test_sender))
.addMessage(getString(R.string.test_message_chat2), System.currentTimeMillis(), getString(R.string.test_sender)))
}
以上demo已上传至:https://github.com/andrsay/OreoNotificationSample