Android 之 Notification 必须掌握知识点

SamuelMccar 8年前
   <p><strong>创建并发送一个系统通知</strong></p>    <p>activity_main.xml</p>    <pre>  <code class="language-java"><?xml version="1.0" encoding="utf-8"?>  <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"      xmlns:tools="http://schemas.android.com/tools"      android:layout_width="match_parent"      android:layout_height="match_parent"      tools:context="com.example.mu_16jj.notificationdemo.MainActivity">      <TextView          android:id="@+id/tv_send_notification"          android:layout_width="wrap_content"          android:layout_height="35dp"          android:layout_centerInParent="true"          android:gravity="center"          android:background="@color/colorAccent"          android:text="send notification"          android:textSize="16sp" />  </RelativeLayout></code></pre>    <p>很简单的布局,就一个 TextView 利用其点击事件来发送通知。</p>    <p>创建 Notification</p>    <pre>  <code class="language-java">private void createNotification() {          notification = new NotificationCompat.Builder(MainActivity.this)                  .setContentTitle("Notification title")                  .setTicker("Ticker method function...")                  .setContentText("Notification content text")                  .setSubText("subtext...")                  .setWhen(System.currentTimeMillis())                  .setSmallIcon(R.mipmap.ic_launcher_round)                  .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.drawable.large_icon))                  .build();      }</code></pre>    <p>通过 NotificationCompat.Build 来构建一个 Notification 对象,并设置一系列属性(每个属性对应的效果后面展示)。</p>    <p>发送按钮监听</p>    <pre>  <code class="language-java">notificationManager.notify(1, notification);</code></pre>    <p>这里需要说明的是,Notification 的发送还是由 NotificationManager 来管理的,第一个参数用来标识一个唯一的 Notification;第二个参数就是需要发送的 Notification 对象。我们在 onCreate 方法中通过以下代码初始化了通知管理器对象:</p>    <pre>  <code class="language-java">notificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);</code></pre>    <p>运行效果</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/d612210c7d59ee348da6cc33bda54c2c.png"></p>    <p>通知效果</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/cb34b106a791db97c30d1ed66135e397.png"></p>    <p>属性介绍</p>    <p>创建通知时的一些基本属性就利用效果图解释完了,其中的 setTicker 需要真机运行才可以看到效果。</p>    <p>可是,这个时候我们点击这条通知后,该通知一直在系统状态栏,既没有消失,也没有页面跳转,这并不是我们想要的效果,所以下面就来实现点击通知跳转页面,这里以系统提供的登录 Activity 为目的界面。</p>    <ul>     <li> <p><strong>实现点击跳转界面</strong></p> </li>    </ul>    <p>修改我们创建通知的方法</p>    <pre>  <code class="language-java">private void createNotification() {          Intent intent = new Intent(this, LoginActivity.class);          PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);          notification = new NotificationCompat.Builder(MainActivity.this)                  .setContentTitle("Notification title")                  .setContentText("Notification content text")                  .setSubText("subtext...")                  .setWhen(System.currentTimeMillis())                  .setSmallIcon(R.mipmap.ic_launcher_round)                  .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.drawable.large_icon))                  .setContentIntent(pendingIntent)                  .setAutoCancel(true)                  .build();      }</code></pre>    <p>可以看到,这里增加了一个 setContentIntent 方法,该方法接收一个 PendingIntent 参数,而获取该实例则可以有三种途径:getActivity(),getBroadcast(),getService();这几个方法所接收的参数都是相同的,第一个是上下文,第二个参数一般很少使用,传 0 即可,第三个参数便是点击通知需要跳转的 Intent 意图对象(LoginActivity直接使用 AS 提供的模板),第四个参数用于确定 PendingIntent 的行为,有 4 种植可选:FLAG_ONE_SHOT,FLAG_NO_CREATE,CANCEL_CURRENT 和 FLAG_UPDATE_CURRENT。</p>    <p>运行效果</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/683f5c94534c72b5a469003355e44aa2.gif"></p>    <p><strong>主要常量</strong></p>    <p>FLAG_CANCEL_CURRENT:如果当前系统中已经存在一个相同的 PendingIntent 对象,那么就将先前已有的 PendingIntent 取消,然后重新生成一个 PendingIntent 对象。</p>    <p>FLAG_NO_CREATE:如果当前系统中不存在相同的 PendingIntent 对象,系统将不会创建该 PendingIntent 对象而是直接返回 null。</p>    <p>FLAG_ONE_SHOT:该 PendingIntent 只作用一次。在该 PendingIntent 对象通过send() 方法触发过后,PendingIntent 将自动调用 cancel() 进行销毁,那么如果你再调用 send() 方法的话,系统将会返回一个 SendIntentException。</p>    <p>FLAG_UPDATE_CURRENT:如果系统中有一个和你描述的 PendingIntent 对等的PendingInent,那么系统将使用该 PendingIntent 对象,但是会使用新的 Intent 来更新之前 PendingIntent 中的 Intent 对象数据,例如更新 Intent 中的 Extras。</p>    <p><strong>PendingIntent 和 Intent</strong></p>    <p>PendingIntent 是一个特殊的 Intent,主要区别是 intent 是立马执行,PendingIntent 是待确定的 Intent。PendingIntent 的操作实际上是传入的 intent 的操作。使用 pendingIntent 的目的主要是用于所包含的 intent 执行是否满足某些条件。</p>    <ul>     <li><strong>实现点击通知后,这条通知从系统状态栏消失</strong></li>    </ul>    <p>有两种办法,一种是给 Notification 设置</p>    <pre>  <code class="language-java">setAutoCancel(true)</code></pre>    <p>另一种是通过通知管理器调用 cancel 方法,该方法有一个参数,就是我们前面发送的时候设置的唯一标识。</p>    <p><strong>通知,设置特效</strong></p>    <p>我们常见的手机 APP 当有推送消息或者电话之类的通知时,我们的手机 LED 灯会亮起来,而且颜色也会不一样,有时候还会有声音,或者震动之类的,下面就来学习这些方法,这里我通过监听手机锁屏的广播来发送通知(效果会更好),不再使用上面的按钮来发送(锁屏速度快的话效果还是可以看到的)。</p>    <p>创建通知代码</p>    <pre>  <code class="language-java">private void createNotification() {          Intent intent = new Intent(this, LoginActivity.class);          PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, 0);          notification = new NotificationCompat.Builder(MainActivity.this)                  .setContentTitle("Notification title")                  .setContentText("Notification content text")                  .setSubText("subtext...")                  .setWhen(System.currentTimeMillis())                  .setSmallIcon(R.mipmap.ic_launcher_round)                  .setLargeIcon(BitmapFactory.decodeResource(getResources(), R.drawable.large_icon))                  // 跳转界面 系统状态栏通知消失                  .setContentIntent(pendingIntent)                  .setAutoCancel(true)                  // 收到通知时播放声音并震动                  .setDefaults(NotificationCompat.DEFAULT_SOUND) // 默认通知声音  //                .setSound(Uri.fromFile(new File("/system/media/audio/ringtones/Luna.ogg")))                  .setVibrate(new long[]{0, 1000, 1000, 1000})                  // LED 灯颜色                  .setLights(Color.RED, 1000, 1000)                  // 根据手机的当前环境使用默认  //                .setDefaults(NotificationCompat.DEFAULT_ALL)                  .build();      }</code></pre>    <p>这里的注释已经很清楚了,需要注意的是震动是需要权限的:</p>    <pre>  <code class="language-java"><uses-permission android:name="android.permission.VIBRATE" /></code></pre>    <p>广播接收器</p>    <pre>  <code class="language-java">/**       * 监听手机锁屏       */      public class MyReceiver extends BroadcastReceiver {            @Override          public void onReceive(Context context, Intent intent) {              if (intent.getAction().equals(Intent.ACTION_SCREEN_OFF)) {                  notificationManager.notify(1, notification);              }          }      }</code></pre>    <p>关注注册广播问题,请看《 <a href="/misc/goto?guid=4959747034134038084" rel="nofollow,noindex"> <strong>Android 广播介绍以及使用示例</strong> </a> 》</p>    <ol>     <li> <p>高级之 setStyle() 和 setPriority() 方法</p>      <ul>       <li> <p>通知内容支持长文本</p> <pre>  <code class="language-java">.setStyle(new NotificationCompat.BigTextStyle().bigText("The failures and reverses which await men - and one after another sadden the brow of youth - add a dignity to the prospect of human life, which no Arcadian success would do. -- Henry David Thoreau"))</code></pre> <p>运行效果</p> </li>      </ul> </li>    </ol>    <p style="text-align:center"><img src="https://simg.open-open.com/show/c6b4fdddebfb36d4da461b03b849271a.png"></p>    <ul>     <li>通知内容支持大图片 <pre>  <code class="language-java">.setStyle(new NotificationCompat.BigPictureStyle().bigPicture(BitmapFactory.decodeResource(getResources(), R.drawable.large_icon)))</code></pre> 运行效果</li>    </ul>    <p style="text-align:center"><img src="https://simg.open-open.com/show/ab296dc0e2d3f21700cb3dd4f5b0befd.png"></p>    <ul>     <li> <p>通知优先级</p> <p>通知的优先级共有五个常量值可选:PRIORITY_DEFAULT(默认);PRIORITY_MIN 最低优先级,系统可能在特定场景才会显示这条通知,比如用户下拉状态栏;PRIORITY_LOW 较低优先级,系统可能会将这类通知缩小,或改变其显示的顺序,将其排在更重要的通知之后;PRIORITY_HIGH 表示较高优先级,系统可能会将这类通知放大、改变其显示的顺序,将其排在靠前的位置; PRIORITY_MAX 表示最高的重要通知,这类通知必须要让用户立刻看到,甚至需要用户做出响应操作。</p> <pre>  <code class="language-java">.setPriority(NotificationCompat.PRIORITY_MAX)</code></pre> </li>    </ul>    <p>运行效果(PRIORITY_MAX)</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/9ae43cdf27851431d83b942d25af1057.png"></p>    <p><strong>自定义布局通知</strong></p>    <p>我们在已有的界面基础上再增加一个 TextView 用它的点击事件来发送自定义通知,自定义通知代码如下:</p>    <pre>  <code class="language-java">/**       * 创建自定义布局通知       */      private void createCustomNotification() {          RemoteViews remoteViews = new RemoteViews(getPackageName(), R.layout.custom_notification_layout);          remoteViews.setImageViewResource(R.id.iv_cus_notification, R.mipmap.ic_launcher);          remoteViews.setTextViewText(R.id.tv_cus_notification, "Notification of Android");          remoteViews.setTextViewText(R.id.btn_cus_notification, "Button");            cusNotification = new NotificationCompat.Builder(MainActivity.this)                  // 切记,一定要设置,否则通知显示不出来                  .setSmallIcon(R.mipmap.ic_launcher)                  .setContent(remoteViews)                  .setTicker("Custom Notification")                  .setAutoCancel(true)                  .setDefaults(NotificationCompat.DEFAULT_ALL)                  .build();      }</code></pre>    <p>可以看到,相比系统通知的创建,这里多了一个新的 API——RemoteViews. 我们的自定义布局就是通过它来加载,通过它的一系列 setXXX 方法可以给我们的布局中控件设置相关属性,然后通过 Builder 的 setContent 方法将其设置上去,这样我们的自定义通知效果就实现了。</p>    <p>点击事件</p>    <pre>  <code class="language-java">notificationManager.notify(2, cusNotification);</code></pre>    <p>运行效果</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/f2dcb75609967082efd4c8b9b694e23a.png"></p>    <p>关于通知的自定义布局从效果就可以看到,很简单,这里不贴出。</p>    <p><strong>注意点</strong></p>    <p>自定义通知布局的可用高度取决于通知视图。普通视图布局限制为 64 dp,扩展视图布局限制为 256 dp。</p>    <p>无论是系统通知还是自定义布局通知,smallIcon 是必须设置的。</p>    <p> </p>    <p> </p>    <p>来自:https://juejin.im/post/58ec1c210ce463006babbba8</p>    <p> </p>