美文网首页
Android 8.0系统的通知栏适配

Android 8.0系统的通知栏适配

作者: coke613 | 来源:发表于2019-03-04 16:33 被阅读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

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

    相关文章

      网友评论

          本文标题:Android 8.0系统的通知栏适配

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