在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_DEFAULTif (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系统 小米/三星/索尼 应用启动图标未读消息数(BadgeNumber)动态提醒相关推荐

  1. 应用启动图标未读消息数显示 工具类

    /* * 应用启动图标未读消息数显示 工具类 (效果如:QQ.微信.未读短信 等应用图标) * */ public class BadgeUtil { /** * Set badge count * ...

  2. Android系统 应用图标显示未读消息数(BadgeNumber) 桌面app图标的角标显示

    转载请标明出处:http://blog.csdn.net/xx326664162/article/details/51082574 文章出自:薛瑄的博客 你也可以查看我的其他同类文章,也会让你有一定的 ...

  3. android图标未读消息,Android系统 应用图标显示未读消息数(BadgeNumber) 桌面app图标的角标显示...

    原理 添加角标的原理就是发送一个Broadcast(广播),在广播的Intent中指定需要被添加角标的应用的packageName(包名),className(类名),count(角标数目).当然了, ...

  4. Android系统 应用图标显示未读消息数(BadgeNumber) 桌面app图标的角标显示

    http://www.51itong.net/android-badgenumber-9789.html

  5. Android app图标显示未读消息数

    转载请标明出处:http://blog.csdn.net/xx326664162/article/details/51082574 文章出自:薛瑄的博客 你也可以查看我的其他同类文章,也会让你有一定的 ...

  6. android底部导航栏带消息数的框架,GitHub - BarkSheep/Android-NavMenuLayout: 一个底部导航栏, 实现了显示未读消息数, 显示红点等效果的封装...

    Android-NavMenu-master 一个底部导航栏, 实现了显示未读消息数, 显示红点等效果的封装. 添加依赖 1. 在项目根目录的 build.gradle 中添加 allprojects ...

  7. android仿微信实现消未读消息数在tab栏提醒

    最近公司有个项目,要仿照微信的未读消息数提醒,提醒用户通知公告的未读数,找了几个都不太好用,后来找前辈咨询,给了一个库,感觉特别实用,在这里分享给大家. 首先给大家看项目的原型图截图: 然后是代码实现 ...

  8. android 未读信息反复提醒,Android仿微信未读消息数提示显示数字BadgeView大于99条显示99+...

    [实例简介] Android仿微信未读消息数提示显示数字BadgeView大于99条显示99+ [实例截图] [核心代码] BadgeView └── BadgeView ├── app │   ├─ ...

  9. android+仿qq未读消息数量,仿qq自定义未读消息数显示角标

    66FF020E13B921CB19C7542F4801AF43.png 如图所示,我们需要实现的效果 在消息tab上,是一个组合的自定义view,具体实现如下 android:layout_widt ...

最新文章

  1. nginx 反向代理,动静态请求分离,proxy_cache缓存及缓存清除
  2. 计算机病毒按破坏性分为哪两类,计算机导论复习要点.doc
  3. TPYBoard:一款可以发挥无限创意的MicroPython开发板
  4. 不安装游戏apk直接启动法
  5. dsh批量管理linux服务器
  6. 教育|俄罗斯天才少女200万年薪加入华为,22岁拿下世界编程冠军,同天队友也宣布加盟...
  7. 《htmlxhtml权威指南》部分标签语义学习
  8. php有的图片显示不出来,图片显示不出来,但是数据库里有显示
  9. 三行代码让你轻松下载全网任意视频-Python小知识
  10. 什么思维是计算机科学的基础概念,计算思维基础
  11. php过滤文本中的手机号,座机号,qq,邮箱地址
  12. 腾讯/京东/网易校招笔试刷人超70%,这份笔试自救指南请收好
  13. linux 无线ap模式,在Linux(CentOS)上开启小度WIFI无线AP模式
  14. BOOST元状态机用户手册之三教程(Meta State Machine (MSM))(1)——基本前端及例程
  15. 【吐槽大会】互联网大厂秋招版
  16. 库卡(KUKA)机器人入门学习必备知识
  17. Python爬猫眼电影影评及可视化 Robin NJU
  18. python内存清理_python如何清理内存
  19. A项目之三:价格与销量分布直方图
  20. SaaS应用加速,助力企业应对业务挑战,高效运转

热门文章

  1. 枚举方法在swtch中的用法
  2. 一个人年轻的时候,总以为疾病和死神只会光顾别人
  3. 大数据茶馆-决策树系列
  4. Springboot Vue个人简历网站系统java项目源码
  5. 公积金查询,公积金账号查询
  6. [Kaggle]泰坦尼克号沉没预测
  7. linux之域名更改
  8. 【论文笔记】Disentangled Graph Collaborative Filtering --- SIGIR2020
  9. 爱库存low在哪里?5毛钱一斤的服装尾货,一年卖出30亿!
  10. 从unity到unreal4 虚幻4学习笔记 一 : ue4与vs studio、vs code开发环境的安装与配置