创建状态栏通知 notification

    技术2022-05-19  29

    懒骨头(http://blog.csdn.com/iamlazybone)

    一个状态栏通知,会在系统的状态栏添加一个图标,并且在状态窗口添加一条信息。当用户点击这个信息时,android会发送一个intent请求,通常是启动一个已定义的activity。你可以添加声音、震动、闪屏给设备来提醒用户。

    通常一个后台服务运行时,如果需要提醒用户一些事件、或者让用户反馈一些信息时,通常用到状态栏提醒。一个后台Service永远不会自己运行一个activity来接受用户交互,一般的,后台服务会添加一个状态栏通知来与用户进行交互。

    下图显示了一个状态栏通知:

     

    接下来的会显示状态栏窗口。用户可以打开这个窗口通过下滑状态栏。或者使用menu菜单键选择。

     

    --------------------------------------------------------------------------------

    基础知识

    一个activity或者Service可以初始化状态栏通知,因为activity只有在活动状态下才能执行一些命令,所以你需要从一个service来建立状态通知。当用户启动了其他程序或者设备已经休眠时,通过这种方式,通知就可以在后台被创建。你要用到这两个类:Notification类和NotificationManager类。

    Notification类来定义状态通知的属性,比如图标,提示信息,或者提示声音。NotificationManager是一个android系统的服务,来管理和运行所有通知的,他不能被实例化,你可以用getSystemService()方法获得他的句柄。当你想通知用户时,调用notify()方法即可。

    创建一个菜单栏通知:

    1-获得MotificationManager的引用。

    view plaincopy to clipboardprint?01.String ns = Context.NOTIFICATION_SERVICE;   02.NotificationManager mNotificationManager = (NotificationManager) getSystemService(ns);  String ns = Context.NOTIFICATION_SERVICE;NotificationManager mNotificationManager = (NotificationManager) getSystemService(ns);

    2-实例化Notification:

    view plaincopy to clipboardprint?01.int icon = R.drawable.notification_icon;   02.CharSequence tickerText = "Hello";   03.long when = System.currentTimeMillis();   04.Notification notification = new Notification(icon, tickerText, when);  int icon = R.drawable.notification_icon;CharSequence tickerText = "Hello";long when = System.currentTimeMillis();Notification notification = new Notification(icon, tickerText, when);

    3-定义Notification,如显示icon、目标intent等信息

    view plaincopy to clipboardprint?01.Context context = getApplicationContext();   02.CharSequence contentTitle = "My notification";   03.CharSequence contentText = "Hello World!";   04.Intent notificationIntent = new Intent(this, MyClass.class);   05.PendingIntent contentIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);   06.notification.setLatestEventInfo(context, contentTitle, contentText, contentIntent);  Context context = getApplicationContext();CharSequence contentTitle = "My notification";CharSequence contentText = "Hello World!";Intent notificationIntent = new Intent(this, MyClass.class);PendingIntent contentIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);notification.setLatestEventInfo(context, contentTitle, contentText, contentIntent);

    4-传递给Manager.

    view plaincopy to clipboardprint?01.private static final int HELLO_ID = 1;   02.mNotificationManager.notify(HELLO_ID, notification);  private static final int HELLO_ID = 1;mNotificationManager.notify(HELLO_ID, notification);

    好了,一个通知写完了。

    --------------------------------------------------------------------------------

    管理你的通知

    NotificationManager是一个管理所有通知的系统服务。你可以通过getSystemService()方法得到她的句柄:

    view plaincopy to clipboardprint?01.String ns = Context.NOTIFICATION_SERVICE;   02.NotificationManager mNotificationManager = (NotificationManager) getSystemService(ns);  String ns = Context.NOTIFICATION_SERVICE;NotificationManager mNotificationManager = (NotificationManager) getSystemService(ns);

    当你想发送一个状态栏通知时,通过notify(int,Notification)方法传递Notification对象给NotificationManager,第一个参数是Notification的id,第二个参数是通知的对象。id是一个唯一的标示,当你改变通知或者进行一些的活动时必须要用到它。

    当用户从通知窗口选择了某个通知时,会给这个通知添加一个“FLAG_AUTO_CANCEL”标记来移除这个通知。你也可以使用cancel(int)方法,通过指定通知id来取消某个通知,或者干脆cancelAll()。

    --------------------------------------------------------------------------------

     建立一个通知

    一个通知对象定义了显示在状态栏的一些细节信息,和所有通知方式,例如声音、闪光。

    一个状态栏通知必须包括以下几点:

    @ 显示在状态栏的图标

    @ 一个标题(除非你自定义了提示界面)

    @ 一个Pending Intent,即点击后要做的操作。

    可选项包括以下:

    @ 状态栏提示文本

    @ 提示声音

    @ 震动

    @ 闪光

    初学者套件?为新的通知包含了MNotification(int,CharSequence,long)构造方法和setLatestEventInfo(Context,CharSequence,CharSequence,PendingIntent)方法。这些参数是一个通知的必要参数。下面的代码片段是一个简单的例子:

    view plaincopy to clipboardprint?01.int icon = R.drawable.notification_icon;        // icon from resources   02.CharSequence tickerText = "Hello";              // ticker-text   03.long when = System.currentTimeMillis();         // notification time   04.Context context = getApplicationContext();      // application Context   05.CharSequence contentTitle = "My notification";  // expanded message title   06.CharSequence contentText = "Hello World!";      // expanded message text   07.Intent notificationIntent = new Intent(this, MyClass.class);   08.PendingIntent contentIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);   09.// the next two lines initialize the Notification, using the configurations above   10.Notification notification = new Notification(icon, tickerText, when);   11.notification.setLatestEventInfo(context, contentTitle, contentText, contentIntent);  int icon = R.drawable.notification_icon;        // icon from resourcesCharSequence tickerText = "Hello";              // ticker-textlong when = System.currentTimeMillis();         // notification timeContext context = getApplicationContext();      // application ContextCharSequence contentTitle = "My notification";  // expanded message titleCharSequence contentText = "Hello World!";      // expanded message textIntent notificationIntent = new Intent(this, MyClass.class);PendingIntent contentIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);// the next two lines initialize the Notification, using the configurations aboveNotification notification = new Notification(icon, tickerText, when);notification.setLatestEventInfo(context, contentTitle, contentText, contentIntent);

    更新通知

    你可以在事件改变时修改状态栏的通知。例如,当有一条未读短信时又来了一条短信,此时状态栏应该显示新的短信。像这种情况下,添加一条洗得通知不如更改原有通知更加合理,因为后者能避免通知的混乱。

    因为每个通知都有一个唯一的Id,你可以通过setLatestEventInfo()方法修改通知,然后调用notify()让其显示。

    你可以修改每个属性,除了context和下拉状态栏时显示的标题和文本。你可以通过setLatestEventInfo()方法给contentTitle和conTentText设置文本信息,然后调用notify()方法来更新通知。(当然了你可以创建自己的布局文件,那样的话更新显示文本就没有效果了)。

    添加声音

    你可以使用默认的声音来提醒用户有一个新通知。方法是添加一个“DEFAULT_SOUND”参数。

    view plaincopy to clipboardprint?01.notification.defaults |= Notification.DEFAULT_SOUND;  notification.defaults |= Notification.DEFAULT_SOUND;

    如果使用非默认的声音,需要传递一个URI资源引用。例如:

    view plaincopy to clipboardprint?01.notification.sound = Uri.parse("file:///sdcard/notification/ringer.mp3");  notification.sound = Uri.parse("file:///sdcard/notification/ringer.mp3");

    下个例子中,音频文件从MediaStore类中获取:

    view plaincopy to clipboardprint?01.notification.sound = Uri.withAppendedPath(Audio.Media.INTERNAL_CONTENT_URI, "6");  notification.sound = Uri.withAppendedPath(Audio.Media.INTERNAL_CONTENT_URI, "6");

    这种情况下,资源id为6的音频文件时已知的,并且已添加到content的URI中。如果你不知道exact ID,你必须在MediaStore中查询所有可用的资源。参考 Content Providers文档。

    如果你想让提示声音一直播放知道用户响应位置,你可以添加“FLAG_INSISTENT”参数。

    注意:如果包含“DEFAULT_SOUND”参数,那么默认声音会覆盖其他的声音设置。

    添加震动

    你可以使用默认的震动方式提示用户:

    view plaincopy to clipboardprint?01.notification.defaults |= Notification.DEFAULT_VIBRATE;  notification.defaults |= Notification.DEFAULT_VIBRATE;

     或者使用自己的振动方式,例如:

    view plaincopy to clipboardprint?01.long[] vibrate = {0,100,200,300};   02.notification.vibrate = vibrate;  long[] vibrate = {0,100,200,300};notification.vibrate = vibrate;

     这个数组定义了交替的震动和关闭,一毫秒为单位。第一个值是等待多久开始震动,第二个值是第一次震动的时间,第三个是停止震动的时间,以此类推。定义多长时间都行,但是不能设置为重复。

    注意:如果设置了默认,则其他振动方式无效。

    添加手机灯闪烁

    让手机闪动LED灯,你可以实现默认的闪动色绘制。或者定义自己的闪动效果。

    使用默认的闪光效果,代码如下:

    view plaincopy to clipboardprint?01.notification.defaults |= Notification.DEFAULT_LIGHTS;  notification.defaults |= Notification.DEFAULT_LIGHTS;

    定义自己的效果,可以设置颜色、定义显示的时间,一些常用的值如下:

    view plaincopy to clipboardprint?01.notification.ledARGB = 0xff00ff00;   02.notification.ledOnMS = 300;   03.notification.ledOffMS = 1000;   04.notification.flags |= Notification.FLAG_SHOW_LIGHTS;  notification.ledARGB = 0xff00ff00;notification.ledOnMS = 300;notification.ledOffMS = 1000;notification.flags |= Notification.FLAG_SHOW_LIGHTS;

    在这个例子里,绿灯先显示300毫秒然后关闭一秒钟,设备不是支持所有的颜色,而且不是所有的设备都支持相同颜色。所以设备会按照指定值显示最接近的颜色。绿色是常见的颜色。

    通知的更多特性

    你可以添加几个特性到你的通知中去。下面是一些有用的设置:

    "FLAG_AUTO_CANCEL"标记

    这个标记会在用户选择查看通知窗口后自动关闭通知。

    "FLAG_INSISTENT"

    在用户响应之前一直重复

    "FLAG_ONGOING_EVENT"

    添加这个标记把通知分在“正在运行”组中,说明程序正在运行,或者后台运行,甚至当程序不可见时,例如播放音乐或者接听电话时。

    "FLAG_NO_CLEAR"

    这个标记让你的通知不会被Clear按钮所取消。对一直进行中的通知非常有用。

    number field

    标记表示当前通知所代表的事件数。这个数字显示在icon之上。如果你打算这样使用,当通知第一次建立时必须从1开始计数。如果你改变了他的值从0或其他值开始,那么他将不会显示。

    iconLevel field

    这个值指当前icon的LevelListDrawable等级。通过改变这个值,你可以让状态栏的icon显示动画。

    更多详细的特性和使用方法餐卡Notification类。

    --------------------------------------------------------------------------------

    建立一个自定义的通知视图

     

    默认情况下,下拉的Notification窗口包含一个标题和文本信息。setLatestEventInfo()有两个默认的参数contentTitle和contentText 。然而你通过RemoteViews也可以定义自己下拉通知视图。上卖弄的截图就显示了一个自定义布局的通知视图,包括一个TextView和ImageView。

    定义自己的通知视图,实例化RemoteViews对象并且传递给contentView。给contentIntent字段传递PendingIntent值。创建一个自定义的通知视图最好的理解方法就是写一个例子:

    1-建立xml布局文件。例如:custom_notification_layout.xml:

    view plaincopy to clipboardprint?01.<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  02.              android:orientation="horizontal"  03.              android:layout_width="fill_parent"  04.              android:layout_height="fill_parent"  05.              android:padding="3dp"  06.              >  07.    <ImageView android:id="@+id/image"  08.              android:layout_width="wrap_content"  09.              android:layout_height="fill_parent"  10.              android:layout_marginRight="10dp"  11.              />  12.    <TextView android:id="@+id/text"  13.              android:layout_width="wrap_content"  14.              android:layout_height="fill_parent"  15.              android:textColor="#000"  16.              />  17.</LinearLayout>  <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"              android:orientation="horizontal"              android:layout_width="fill_parent"              android:layout_height="fill_parent"              android:padding="3dp"              >    <ImageView android:id="@+id/image"              android:layout_width="wrap_content"              android:layout_height="fill_parent"              android:layout_marginRight="10dp"              />    <TextView android:id="@+id/text"              android:layout_width="wrap_content"              android:layout_height="fill_parent"              android:textColor="#000"              /></LinearLayout>

    这是一个自定义的扩展通知的视图,但是ImageView和TextView仍然需要程序定义。RemoteViews提供一些方便的方法来让你定义content。

    2-下面的代码,使用RemoveViews 的方法定义image和text。然后传递RemoteViews 对象给通知的contentView字段。如下:

    view plaincopy to clipboardprint?01.RemoteViews contentView = new RemoteViews(getPackageName(), R.layout.custom_notification_layout);   02.contentView.setImageViewResource(R.id.image, R.drawable.notification_image);   03.contentView.setTextViewText(R.id.text, "Hello, this message is in a custom expanded view");   04.notification.contentView = contentView;  RemoteViews contentView = new RemoteViews(getPackageName(), R.layout.custom_notification_layout);contentView.setImageViewResource(R.id.image, R.drawable.notification_image);contentView.setTextViewText(R.id.text, "Hello, this message is in a custom expanded view");notification.contentView = contentView;

    就像显示的那样,传递程序的包名,布局的资源id给RemoteViews的构造方法。然后,定义ImageView和TextView的content,使用setImageViewResource()和setTextViewText().这种情况下,传递你要设置的view对象的引用id。最后把RemoteViews 对象传递给contentView。

    3-因为你不需要setLatestEventInfo()了,你必须为通知定义一个intent,例如:

    view plaincopy to clipboardprint?01.Intent notificationIntent = new Intent(this, MyClass.class);   02.PendingIntent contentIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);   03.notification.contentIntent = contentIntent;  Intent notificationIntent = new Intent(this, MyClass.class);PendingIntent contentIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);notification.contentIntent = contentIntent;

    4-通知可以这样调用:

    view plaincopy to clipboardprint?01.mNotificationManager.notify(CUSTOM_VIEW_ID, notification);  mNotificationManager.notify(CUSTOM_VIEW_ID, notification);

    RemoteViews 类也包含一些方法让你轻松地添加Chronometer或者ProgressBar在你的通知view里。

    注意:当建立一个自定义的通知view,你必须特别小心自定义的布局来适用不同设备的分辨率。尤其这种情况下,更要特别的小心,过于复杂的布局一定要多多测试

    本文来自博客,转载请标明出处:http://blog.csdn.net/iamlazybone/archive/2010/10/22/5959598.aspx


    最新回复(0)