Android之Notification和Remoteview

lvxiangan發表於2019-01-03

Notification是通過建造者模式來建立。
為了相容低版本,v4 Support Library中提供了NotificationCompat.Builder()這個替代方法。
它與原來的Notification.Builder()類似,二者沒有太大區別。

 

通知基本用法

先來一段程式碼demo:
NotificationCompat.Builder builder = new NotificationCompat.Builder(context); // 舊API
Intent intent = new Intent(this, MainActivity.class);  //點選之後進入MainActivity
PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
Notification notification = builder.setSmallIcon(R.mipmap.ic_launcher)  //設定小圖示
                .setTicker("hello world")  //設定文字
                .setWhen(System.currentTimeMillis())  //通知的時間
                .setAutoCancel(true)  //點選後消失
                .setContentIntent(pendingIntent)  //設定意圖
                .build();  //建立通知物件完成
NotificationManager manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
manager.notify(1, notification);  //顯示通知


屬性

  • setSmallIcon 小圖示
  • setContentTitle 標題
  • setContentText 詳細資訊
  • setAutoCancel(boolean) 是否點選通知自動取消
  • setOngoing(boolean) 設定是否不能消除該通知; 利用它設定他為一個正在進行的通知。他們通常是用來表示一個後臺任務,使用者積極參與(如播放音樂)或以某種方式正在等待,因此佔用裝置(如一個檔案下載,同步操作,主動網路連線)
  • setWhen() 時間,如果不設定,則預設顯示當前的系統時間
  • setContentIntent(pendingIntent) 設定意圖
  • setPriority(Notification.PRIORITY_DEFAULT) 設定該通知優先順序
  • setDefaults 向通知新增聲音、閃燈和震動效果,最簡單、最一致的方式是使用當前的使用者預設設定,使用defaults屬性
Notification.DEFAULT_VISIBLE //新增預設震動提醒 需要VIBRATE permission
Notification.DEFAULT_SOUND //新增預設聲音提醒
Notification.DEFAULT_LIGHTS //新增預設三色燈提醒
Notification.DEFAULT_ALL //新增預設以上三種全部提醒
  • setUsesChronometer 是否顯示時間計時
  • setProgress(intmax, int progress, boolean indeterminate) 設定進度, indeterminate 表示是否是不明確的進度條。可以通過setProgress(0,0,false)移除進度條
  • setContentIntent(PendingIntent intent) 功能:設定點選通知欄意圖。如點選跳轉到一個activity
  • addAction 向通知新增操作,操作通常與通知的content相連被系統作為按鈕來顯示。在系統的content下方顯示圖片與title,點選這個圖片或者title就會觸發設定的intent
  • setLargeIcon(BitmapFactory.decodeResource(getResources(),R.drawable.wechat));

 

PendingIntent

  • FLAG_CANCEL_CURRENT:如果要建立的PendingIntent已經存在了,那麼在建立新的PendingIntent之前,原先已經存在的PendingIntent中的intent將不能使用
  • FLAG_NO_CREATE:如果要建立的PendingIntent尚未存在,則不建立新的PendingIntent,直接返回null
  • FLAG_ONE_SHOT:相同的PendingIntent只能使用一次,且遇到相同的PendingIntent時不會去更新PendingIntent中封裝的Intent的extra部分的內容
  • FLAG_UPDATE_CURRENT:如果要建立的PendingIntent已經存在了,那麼在保留原先PendingIntent的同時,將原先PendingIntent封裝的Intent中的extra部分替換為現在新建立的PendingIntent的intent中extra的內容

 

如何自定義通知的佈局?

需要用到RemoteViews。我們先寫好自定義通知的佈局檔案:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="48dp"
    android:orientation="vertical"
    android:paddingEnd="48dp"
    android:paddingStart="48dp">
    <ImageView
        android:id="@+id/icon_5"
        android:layout_width="48dp"
        android:layout_height="48dp"
        android:layout_centerInParent="true"
        android:src="@mipmap/ic_launcher_round" />
    <TextView
        android:id="@+id/tv_5_s"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:layout_alignParentStart="true"
        android:gravity="center_vertical"
        android:text="Left" />
    <TextView
        android:id="@+id/tv_5_e"
        android:layout_width="wrap_content"
        android:layout_height="match_parent"
        android:layout_alignParentEnd="true"
        android:gravity="center_vertical"
        android:text="Right" />
</RelativeLayout>

 

將自定義的佈局運用在通知中

 NotificationCompat.Builder builder = new NotificationCompat.Builder(this);
        Intent intent = new Intent(this, MainActivity.class);
        PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);
        RemoteViews remoteViews = new RemoteViews(getPackageName(), R.layout.layout_notificaiton);
        remoteViews.setTextViewText(R.id.tv_5_s, "Start");//通過id-內容的方式設定remoteview中控制元件的內容,底層實現是通過Binder跨程式通訊
        remoteViews.setTextViewText(R.id.tv_5_e, "End");
        remoteViews.setImageViewResource(R.id.icon_5, R.mipmap.ic_launcher);
        remoteViews.setOnClickPendingIntent(R.id.tv_5_e, pendingIntent);
        Notification notification = builder.setSmallIcon(R.mipmap.ic_launcher)//通知的構建過程基本與預設相同
                .setTicker("hello world")
                .setWhen(System.currentTimeMillis())
                .setAutoCancel(true)
                .setContent(remoteViews)//在這裡設定自定義通知的內容
                .build();
        NotificationManager manager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);
        manager.notify(2, notification);可以看到Remoteview更新內容的方式比較特別,因為不能通過findViewById的方式獲取控制元件(跨程式了獲取不到),所以更新實際上是通過Binder傳送更新資訊到remoteview來更新UI的。

 

Android 8.0(Api 26) 及以上如何使用?

NotificationChannel是android8.0新增的特性,如果App的targetSDKVersion>=26,沒有設定channel通知渠道的話,就會導致通知無法展示。

String channelID = "1";
String channelName = "channel_name";

NotificationChannel channel = new NotificationChannel(channelID, channelName, NotificationManager.IMPORTANCE_HIGH);

NotificationManager manager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE); 

manager.createNotificationChannel(channel);
// 利用channelName可以在系統設定頁面對app某個名稱的通知進行管理
NotificationCompat.Builder builder =new NotificationCompat.Builder(context, channelName);
builder.setContentText(msgDesc);
builder.setContentTitle(msgTitle);

//建立通知時指定channelID
builder.setChannelId(channelID);
Notification notification = builder.build();

 

注意:修改NotificationChannel 屬性,升級app後該修改不生效,必須解除安裝app重新安裝才能生效,原始碼如下:

public void notifyDownloading(long progress, long num, String chName) {
    Notification.Builder  mBuilder = new Notification.Builder(MainActivity.this, TAG );
    NotificationChannel channel = new NotificationChannel(TAG , chName, NotificationManager.IMPORTANCE_HIGH);
    mNotifyManager.createNotificationChannel(channel);
    mBuilder.setSmallIcon(R.drawable.notification_download_icon);
    mBuilder.setProgress((int) num, (int) progress, false);
    mBuilder.setContentInfo(getPercent((int) progress, (int) num));
    mBuilder.setOngoing(true);
    mBuilder.setWhen(System.currentTimeMillis());
    mBuilder.setContentTitle(chName);
    mBuilder.setContentText("download");
    PendingIntent pendIntent = PendingIntent.getActivity(
            MainActivity.this, NOTIFY_ID, getCurActivityIntent(), PendingIntent.FLAG_UPDATE_CURRENT);
    mBuilder.setContentIntent(pendIntent);
    mNotifyManager.notify(NOTIFY_ID, mBuilder.build());
}
這裡將IMPORTANCE_HIGH修改為IMPORTANCE_LOW,通過Android Studio直接安裝,發現修改不生效。
解決方法:修改建立Notification.Builder的id和NotificationChannel的id,修改後程式碼如下:

public void notifyDownloading(long progress, long num, String chName) {
    Notification.Builder mBuilder = new Notification.Builder(MainActivity.this, TAG + System.currentTimeMillis());
    NotificationChannel channel = new NotificationChannel(TAG + System.currentTimeMillis(), chName, NotificationManager.IMPORTANCE_LOW);
    mNotifyManager.createNotificationChannel(channel);
    mBuilder.setSmallIcon(R.drawable.notification_download_icon);
    mBuilder.setProgress((int) num, (int) progress, false);
    mBuilder.setContentInfo(getPercent((int) progress, (int) num));
    mBuilder.setOngoing(true);
    mBuilder.setWhen(System.currentTimeMillis());
    mBuilder.setContentTitle(chName);
    mBuilder.setContentText("download");
    PendingIntent pendIntent = PendingIntent.getActivity(
            MainActivity.this, NOTIFY_ID, getCurActivityIntent(), PendingIntent.FLAG_UPDATE_CURRENT);
    mBuilder.setContentIntent(pendIntent);
    mNotifyManager.notify(NOTIFY_ID, mBuilder.build());
}
通過System.currentTimeMillis()保證每次建立物件的Id不同。

 

 

 

參考:
https://blog.csdn.net/qq_25749749/article/details/80449108 
https://www.jianshu.com/p/ca92797d925a
https://www.jianshu.com/p/33a18f224a60
https://blog.csdn.net/agaghd/article/details/79016812 
https://www.jianshu.com/p/33a18f224a60
https://blog.csdn.net/agaghd/article/details/79016812 

相關文章