Android系統 小米/三星/索尼 應用啟動圖示未讀訊息數(BadgeNumber)動態提醒
在Android手機上,如QQ、微信當有未讀訊息的時候、我們可以看到在應用的啟動圖示的右上角會有一個紅色圈圈、且圈圈裡會動態顯示未讀訊息的數目,如下圖顯示:
那麼該功能是怎麼實現的呢?
在萬能的網際網路搜尋和翻閱了大量相關資料、也請教了一些技術群裡的大咖們。從他們那裡我獲知、提取了一些關鍵詞:第三方控制元件BadgeView(實現應用內的數字提醒)、快捷圖示、Launcher、反射。
零零碎碎的花費了近一天時間、終於算是弄明白了。寫了個demo測試程式 驗證並自測了一下。 demo效果如下所示:
三星Galaxy S4上測試效果如下:
小米手機上測試效果如下:
實現原理:
首先我們要明白 並不是應用本身處理對啟動圖示進行修改、圖示的動態修改的過程主要是在Launcher裡面完成的.在應用安裝,更新,解除安裝的時候,都會有廣播發出,Launcher在LauncherApplication 中註冊廣播,在LauncherModel中處理接收到廣播的訊息,重新載入更新應用資訊(如:應用圖示、文字等)。但是原生的android系統是並不支援該特性的(及不能通過傳送特定的系統廣播 達到動態修改啟動圖示的效果),但是在強大的第三方Android手機廠商(如:三星、小米)的系統原始碼深度定製下、通過修改了Launcher原始碼,增加/註冊了新的廣播接收器用來接收應用傳送來的未讀訊息數廣播,接收到廣播後,系統將未讀訊息的數目顯示事件交給Launcher去處理,呼叫相關方法去重繪應用的icon,最終達到動態更新應用圖示的效果。
在瞭解了實現原理之後、我們大概明白整個流程是這樣的(原生系統除外):
在第三方手機制造商的ROM下、如果修改了Launcher原始碼且支援了上面所說的未讀訊息數廣播的接收、那麼我們只要在應用中傳送一條能讓系統接收的廣播就可以在這種裝置的手機上實現本篇想要達到的效果。
但是第三方手機制造商們的這種廣播的接收的條件肯定是各不相同的、因此最關鍵的就是要知道各手機制造商的這種廣播的Intent接收條件。
幸運的是 在萬能的網際網路上 總能找到你需要的東西,下面封裝了一個工具類 BadgeUtil.java 實現了不同手機制造商的未讀訊息數目廣播。具體程式碼如下:
import java.lang.reflect.Field;
import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.content.pm.ResolveInfo;
import android.os.Build;
import android.widget.Toast;
/**
* 應用啟動圖示未讀訊息數顯示 工具類 (效果如:QQ、微信、未讀簡訊 等應用圖示)<br/>
* 依賴於第三方手機廠商(如:小米、三星)的Launcher定製、原生系統不支援該特性<br/>
* 該工具類 支援的裝置有 小米、三星、索尼【其中小米、三星親測有效、索尼未驗證】
* @author ice_zhengbin@163.com
*
*/
public class BadgeUtil {
/**
* Set badge count<br/>
* 針對 Samsung / xiaomi / sony 手機有效
* @param context The context of the application package.
* @param count Badge count to be set
*/
public static void setBadgeCount(Context context, int count) {
if (count <= 0) {
count = 0;
} else {
count = Math.max(0, Math.min(count, 99));
}
if (Build.MANUFACTURER.equalsIgnoreCase("Xiaomi")) {
sendToXiaoMi(context, count);
} else if (Build.MANUFACTURER.equalsIgnoreCase("sony")) {
sendToSony(context, count);
} else if (Build.MANUFACTURER.toLowerCase().contains("samsung")) {
sendToSamsumg(context, count);
} else {
Toast.makeText(context, "Not Support", Toast.LENGTH_LONG).show();
}
}
/**
* 向小米手機傳送未讀訊息數廣播
* @param count
*/
private static void sendToXiaoMi(Context context, int count) {
try {
Class miuiNotificationClass = Class.forName("android.app.MiuiNotification");
Object miuiNotification = miuiNotificationClass.newInstance();
Field field = miuiNotification.getClass().getDeclaredField("messageCount");
field.setAccessible(true);
field.set(miuiNotification, String.valueOf(count == 0 ? "" : count)); // 設定資訊數-->這種傳送必須是miui 6才行
} catch (Exception e) {
e.printStackTrace();
// miui 6之前的版本
Intent localIntent = new Intent(
"android.intent.action.APPLICATION_MESSAGE_UPDATE");
localIntent.putExtra(
"android.intent.extra.update_application_component_name",
context.getPackageName() + "/" + getLauncherClassName(context));
localIntent.putExtra(
"android.intent.extra.update_application_message_text", String.valueOf(count == 0 ? "" : count));
context.sendBroadcast(localIntent);
}
}
/**
* 向索尼手機傳送未讀訊息數廣播<br/>
* 據說:需新增許可權:<uses-permission android:name="com.sonyericsson.home.permission.BROADCAST_BADGE" /> [未驗證]
* @param count
*/
private static void sendToSony(Context context, int count){
String launcherClassName = getLauncherClassName(context);
if (launcherClassName == null) {
return;
}
boolean isShow = true;
if (count == 0) {
isShow = false;
}
Intent localIntent = new Intent();
localIntent.setAction("com.sonyericsson.home.action.UPDATE_BADGE");
localIntent.putExtra("com.sonyericsson.home.intent.extra.badge.SHOW_MESSAGE",isShow);//是否顯示
localIntent.putExtra("com.sonyericsson.home.intent.extra.badge.ACTIVITY_NAME",launcherClassName );//啟動頁
localIntent.putExtra("com.sonyericsson.home.intent.extra.badge.MESSAGE", String.valueOf(count));//數字
localIntent.putExtra("com.sonyericsson.home.intent.extra.badge.PACKAGE_NAME", context.getPackageName());//包名
context.sendBroadcast(localIntent);
}
/**
* 向三星手機傳送未讀訊息數廣播
* @param count
*/
private static void sendToSamsumg(Context context, int count){
String launcherClassName = getLauncherClassName(context);
if (launcherClassName == null) {
return;
}
Intent intent = new Intent("android.intent.action.BADGE_COUNT_UPDATE");
intent.putExtra("badge_count", count);
intent.putExtra("badge_count_package_name", context.getPackageName());
intent.putExtra("badge_count_class_name", launcherClassName);
context.sendBroadcast(intent);
}
/**
* 重置、清除Badge未讀顯示數<br/>
* @param context
*/
public static void resetBadgeCount(Context context) {
setBadgeCount(context, 0);
}
/**
* Retrieve launcher activity name of the application from the context
*
* @param context The context of the application package.
* @return launcher activity name of this application. From the
* "android:name" attribute.
*/
private static String getLauncherClassName(Context context) {
PackageManager packageManager = context.getPackageManager();
Intent intent = new Intent(Intent.ACTION_MAIN);
// To limit the components this Intent will resolve to, by setting an
// explicit package name.
intent.setPackage(context.getPackageName());
intent.addCategory(Intent.CATEGORY_LAUNCHER);
// All Application must have 1 Activity at least.
// Launcher activity must be found!
ResolveInfo info = packageManager
.resolveActivity(intent, PackageManager.MATCH_DEFAULT_ONLY);
// get a ResolveInfo containing ACTION_MAIN, CATEGORY_LAUNCHER
// if there is no Activity which has filtered by CATEGORY_DEFAULT
if (info == null) {
info = packageManager.resolveActivity(intent, 0);
}
return info.activityInfo.name;
}
}
在啟動的Activity中、傳送未讀訊息數目廣播 和 重置/清除未讀訊息數目廣播 的呼叫如下:
// 傳送未讀訊息數目廣播:count為未讀訊息數目(int型別)
BadgeUtil.setBadgeCount(getApplicationContext(), count);
// 傳送重置/清除未讀訊息數目廣播:
BadgeUtil.resetBadgeCount(getApplicationContext());
資料參考:
http://blog.csdn.net/andylao62/article/details/41794695
http://blog.csdn.net/wx_962464/article/details/37997299
https://github.com/ekinlyw/android-badge
http://www.tuicool.com/articles/JV7vIr
—————————————————————————————————————
如果文章內容對您有幫助, 可以幫 頂 一下,來支援一下哦!
如果您對文章內容有任何疑問或有更好的見解, 歡迎通過留言或發郵件的方式聯絡我:
ice_zhengbin@163.com
如需要轉載,請註明出處,謝謝!!
—————————————————————————————————————
相關文章
- Android應用設定多個啟動圖示,動態列換應用圖示Android
- Flutter 動態更改應用程式啟動圖示Flutter
- Android動態更換應用圖示Android
- Android動態修改應用圖示和名稱Android
- android重新啟動應用程式和重新啟動系統 .Android
- Jenkins實戰之動態替換Android應用圖示JenkinsAndroid
- windows phone 8 鎖屏介面 顯示應用程式的訊息提醒Windows
- Android 12(S) 圖形顯示系統 - SurfaceFlinger的啟動和訊息佇列處理機制(四)Android佇列
- Android應用為啥啟動慢?小米工程師回應Android工程師
- mac系統應用快速啟動工具Mac
- Android 應用啟動流程Android
- Android系統啟動流程(四)Launcher啟動過程與系統啟動流程Android
- android 實現類似qq未讀訊息點選迴圈顯示Android
- Protobuf_動態訊息-反射反射
- 中興&Omdia:5G訊息推動生態系統創新
- Android 系統啟動流程Android
- 【android 7.1.2】系統啟動Android
- Android系統啟動自動開啟mtklogAndroid
- Android應用啟動流程分析Android
- 直播系統搭建,簡單實現Android應用的啟動頁Android
- Android 動態替換手機桌面圖示Android
- 直播原始碼網站,訊息圖示在收到訊息時展示訊息條數原始碼網站
- Android 12(S) 圖形顯示系統 - 示例應用(二)Android
- Win10系統關閉動態磁貼後重新開啟動態顯示的方法Win10
- android系統啟動之PMS啟動原始碼解析Android原始碼
- Android 系統啟動過程Android
- 短視訊軟體開發,ios啟動圖適配和啟動圖示適配iOS
- hook 系統api啟動未註冊ActivityHookAPI
- Android App應用啟動流程(一)AndroidAPP
- Android進階 - 應用啟動分析Android
- Android 7.0 應用啟動流程分析Android
- android 啟動模式應用場景Android模式
- Linux系統啟動流程圖Linux流程圖
- 圖靈書訊動態圖靈
- 圖解 Android 系列(一)揭祕 Android 系統啟動過程圖解Android
- 【iOS】動態更換圖示iOS
- 【iOS】動態修改APP圖示iOSAPP
- 線上直播系統原始碼,簡單實現Android應用的啟動頁原始碼Android