Android之Notification和Remoteview
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
相關文章
- Android 《Notification》Android
- Android 之 Notification 必須掌握知識點Android
- Android系統修改之Notification佈局修改Android
- Notification之 – Android5.0實現原理(一)Android
- Android中的NotificationAndroid
- Android8.1 SystemUI原始碼分析之 Notification流程AndroidSystemUI原始碼
- Android O 新特性 — NotificationAndroid
- Android 8 通知渠道(Notification Channels)Android
- [轉]Android 通知Notification 詳解Android
- Android 深入理解 Notification 機制Android
- 當Notification和Websocket遇到https、httpWebHTTP
- Android程式保活(二):利用 Notification 提升許可權Android
- Android 中 Service+Notification 斷點續傳下載Android斷點
- 【Android】狀態列通知Notification、NotificationManager詳解(轉載)Android
- laravel NotificationLaravel
- Android之android:theme設定在Application 和 Activity的區別AndroidAPP
- Android之Window和彈窗問題Android
- Flutter 通知(Notification)冒泡原理Flutter
- Notification桌面通知實踐
- Android之drawable和mipmap目錄區別Android
- Android之Intent顯示和隱式呼叫AndroidIntent
- User Notification Framework 框架的使用Framework框架
- oreo上的notification詳解
- Android之工程目錄lib和libs區別Android
- Android自定義View之requestLayout方法和invalidate方法AndroidView
- Android自定義View之invalidate方法和postInvalidate方法AndroidView
- Android之Intent的setClass和setClassName的區別AndroidIntent
- RemoteView 遠端控制軟體怎麼樣,有沒有替代品REMView
- Android之android exported="false"作用AndroidExportFalse
- 基於 EasyWechat 和 Laravel notification 傳送微信小程式模板訊息Laravel微信小程式
- Android Studio之Gradle和Gradle外掛的區別AndroidGradle
- Android自定義View之Paint繪製文字和線AndroidViewAI
- Ant design的Notification原始碼分析原始碼
- Android之StrictModeAndroid
- Android之BroadcastReceiverAndroidAST
- Android之FragmentAndroidFragment
- Android之AlarmManagerAndroid
- android開發 之 Bolts-AndroidAndroid