`

Android为应用添加角标(Badge)

阅读更多
1.需求简介

角标是什么意思呢?

看下图即可明了:


可以看到图中的乐购这个app右上角的红色的圆圈,里面有10这个数字的,就是一种角标。

角标,英语是badge,也就是“徽章,像章,奖章; 象征,标记”的意思。

一般来说,应用的角标是用来标记有多少条提醒(Notification)没读(unread),一旦点击提示进应用阅读了,角标也会消失。

2. Android角标起源

角标原本是苹果的iOS中的东西,Android原生并不支持角标,因为Google的意思是让大家用Notification(提示栏)即可,角标实在大有让处女座“跳崖”的风险。幸好我不是...

最近公司的项目中,客户的一个新需求是在我们的加密信息应用上加上角标功能,因为我们的合作伙伴是三星(可以参看我的这篇文章:程序员在法国 | 我被法国国防部盯上了!),因此我就去网上找相关资料。

找的时候,才知道上面所说的Android原生不支持角标一事。不过无妨,厉害的Android第三方厂商可以通过在自定义的Launcher(启动器)中操作来实现添加角标。

我在第一时间当然是去找三星的移动设备如何添加角标,不过却有幸找到了Github上的比较普适的项目。

把我导向Github的自然是Stack Overflow,而把我导向Stack Overflow的就是Google,因此我会说:为什么程序员一定要会用Google和Stack Overflow?

3. 不错的Github项目

一般来说,现在被引用最多的Android添加和去除角标的Github项目是这位中国人写的:https://github.com/leolin310148/ShortcutBadger

这个项目挺不错,虽然更新不是特别勤快,但最近一次更新是在2016年10月31日,也就是两个月前,还可以接受。

《Android群英传》和《Android群英传:神兵利器》的作者 徐宜生 也在自己的Github上建了一个项目:https://github.com/xuyisheng/ShortcutHelper ,挺有意思,里面还有号称“疯狂模式”的为所有在手机桌面上的应用加上99的角标数的功能,当然了,去除的代码也有,不然处女座岂不是要晕了~

添加角标的原理就是发送一个Broadcast(广播),在广播的Intent中指定需要被添加角标的应用的packageName(包名),className(类名),count(角标数目)。当然了,不同厂商的手机的角标操作的Intent的action是不一样的。

因此,我们如果要给自己的手机里的应用添加角标,只需要简单的利用上面两个项目中的代码即可,一般不需要把全部项目搬过来。当然了,如果你要适配所有手机,那么可以全盘引用项目。

比如我要给三星的手机的应用添加角标,那么我只需要做以下的几步即可:

在AndroidManifest.xml中添加读取和写入角标的权限:
<uses-permission android:name="com.sec.android.provider.badge.permission.READ" />
<uses-permission android:name="com.sec.android.provider.badge.permission.WRITE" />

自己写一个类,随便取名字,比如叫做 BadgeUtils,在类中添加如下内容:
public class BadgeUtils {
  private static final String INTENT_ACTION = "android.intent.action.BADGE_COUNT_UPDATE";
  private static final String INTENT_EXTRA_BADGE_COUNT = "badge_count";
  private static final String INTENT_EXTRA_PACKAGENAME = "badge_count_package_name";
  private static final String INTENT_EXTRA_ACTIVITY_NAME = "badge_count_class_name";

  public static void setBadgeCount(Context context, ComponentName componentName, int badgeCount) {
    Intent intent = new Intent(INTENT_ACTION);     
    intent.putExtra(INTENT_EXTRA_BADGE_COUNT, badgeCount);    
    intent.putExtra(INTENT_EXTRA_PACKAGENAME, componentName.getPackageName());    
    intent.putExtra(INTENT_EXTRA_ACTIVITY_NAME, componentName.getClassName());
    context.sendBroadcast(intent);  
  }
}

使用上面的代码时,只需要传入三个参数,也就是:

Context : 应用的Context。简单。
ComponentName :组件名,略有点麻烦。可以这样来获取(applicationContext就是应用的Context) :
applicationContext.getPackageManager()
.getLaunchIntentForPackage(applicationContext.getPackageName())
.getComponent()

badgeCount :角标的数目,例如10。简单。
当然了,如果你不想要传入三个参数这么麻烦,你也可以再写一个方法getLauncherClassName,就只需要传入两个参数即可。BadgeUtils中的代码变为:
public class BadgeUtils {
  private static final String INTENT_ACTION = "android.intent.action.BADGE_COUNT_UPDATE";
  private static final String INTENT_EXTRA_BADGE_COUNT = "badge_count";
  private static final String INTENT_EXTRA_PACKAGENAME = "badge_count_package_name";
  private static final String INTENT_EXTRA_ACTIVITY_NAME = "badge_count_class_name";

  public static void setBadgeCount(Context context, int badgeCount) {
    String launcherClassName = getLauncherClassName(context);
    if (launcherClassName == null) {
      return;
    }

    Intent intent = new Intent(INTENT_ACTION);     
    intent.putExtra(INTENT_EXTRA_BADGE_COUNT, badgeCount);    
    intent.putExtra(INTENT_EXTRA_PACKAGENAME, context.getPackageName());    
    intent.putExtra(INTENT_EXTRA_ACTIVITY_NAME, launcherClassName);

    context.sendBroadcast(intent);  
  }

  private static String getLauncherClassName(Context context) {
    PackageManager pm = context.getPackageManager();

    Intent intent = new Intent(Intent.ACTION_MAIN); 
    intent.addCategory(Intent.CATEGORY_LAUNCHER);

    List<ResolveInfo> resolveInfos = pm.queryIntentActivities(intent, 0);
    for (ResolveInfo resolveInfo : resolveInfos) {
      String pkgName = resolveInfo.activityInfo.applicationInfo.packageName;
      if (pkgName.equalsIgnoreCase(context.getPackageName())) {
        String className = resolveInfo.activityInfo.name;
        return className;
      }
    }
    return null; 
  }
}

使用时传入两个参数即可:

Context : 应用的Context。
badgeCount :角标的数目,例如10。
4. 清除角标

要清除应用的角标就很简单了,给badgeCount传入0即可。

BadgeUtils.setBadgeCount(context,       
context.getPackageManager()               
.getLaunchIntentForPackage(context.getPackageName())                
.getComponent(),        
0);

或者

BadgeUtils.setBadgeCount(context, 0);

5. 小问题纠错

上面的 https://github.com/leolin310148/ShortcutBadger 这个项目中,基本已经包含了大多数可以定制角标的Android生产厂商的添加角标的代码实现,不过它也提到:

三星和LG(这两个难兄难弟)的代码有很多类似,连角标处理的广播的Intent中的action也是一样的,都是:

"android.intent.action.BADGE_COUNT_UPDATE"

但是作者在三星和LG的两个角标操作实现类中写了注释:

// Deprecated, Samsung devices will use DefaultBadger

// Deprecated, LG devices will use DefaultBadger

意思是“三星和LG的实现代码已经Deprecated(失效了),请用DefaultBadger类”。

因此,这两个需要用 https://github.com/leolin310148/ShortcutBadger/blob/master/ShortcutBadger/src/main/java/me/leolin/shortcutbadger/impl/DefaultBadger.java 中的实现:
private static final String INTENT_ACTION = "android.intent.action.BADGE_COUNT_UPDATE";
private static final String INTENT_EXTRA_BADGE_COUNT = "badge_count";
private static final String INTENT_EXTRA_PACKAGENAME = "badge_count_package_name";
private static final String INTENT_EXTRA_ACTIVITY_NAME = "badge_count_class_name";

@Override
public void executeBadge(Context context, ComponentName componentName, int badgeCount) throws ShortcutBadgeException {
  Intent intent = new Intent(INTENT_ACTION);
  intent.putExtra(INTENT_EXTRA_BADGE_COUNT, badgeCount);
  intent.putExtra(INTENT_EXTRA_PACKAGENAME, componentName.getPackageName());
  intent.putExtra(INTENT_EXTRA_ACTIVITY_NAME, componentName.getClassName());

  if (BroadcastHelper.canResolveBroadcast(context, intent)) {
    context.sendBroadcast(intent);
  } else {
    throw new ShortcutBadgeException("unable to resolve intent: " + intent.toString());
  }
}


不过上面的代码有一个小问题,就是那句

if (BroadcastHelper.canResolveBroadcast(context, intent)) {

在有些设备(比如Samsung Galaxy S5)上会抛出异常(Exception),找不到处理"android.intent.action.BADGE_COUNT_UPDATE"这个Intent的BroadcastReceiver,很奇怪。

但有些设备(比如Samsung Galaxy A5)上又运行正常,没有抛出异常。

解决办法是去除这一个检测,把

if (BroadcastHelper.canResolveBroadcast(context, intent)) {
  context.sendBroadcast(intent);
} else {
  throw new ShortcutBadgeException("unable to resolve intent: " + intent.toString());
}

替换为简单的

context.sendBroadcast(intent);

就可以了。

也就是我上面自己实作时的代码。

6. 总结

Android的角标添加和移除毕竟是基于各大手机厂商的Launcher的定制,因此不是正统的Android技巧,随着厂商的Launcher的改变,也许你的代码未来就不一定有用了,因此需要不断修改,“推陈出新”。

不过正所谓“生命在于折腾”,而这也是我们喜欢Android系统的原因。这个萌萌的机器人可以经得起我们随意折腾,在嵌入式领域的应用前途也是很不错的。

大家在平时学习编程的时候,也可以把自己的代码或经验汇总到Github项目,一来惠己利人,二来提高自己的业界知名度。

Git,Github和Gitlab简介和基本使用
Github | 如何贡献Android开源项目和提交补丁
Github改版+我的Github打怪升级之路
  • 大小: 28.9 KB
分享到:
评论

相关推荐

    Android-BadgeNumberManager-为你的应用添加桌面角标的Android库

    通过`BadgeNumberManager`,开发者可以快速且方便地为Android应用添加角标功能,提升用户体验,让用户一眼就能看到应用的状态。在实际开发中,合理利用这个库可以提高开发效率,减少兼容性问题的解决时间。

    Android为应用添加数字角标的简单实现

    总的来说,为Android应用添加数字角标涉及到权限的添加、广播的发送以及兼容性处理。虽然Android系统本身不直接支持角标功能,但通过一些技巧和第三方库,开发者可以轻松地在应用中实现这一功能,提高用户体验。记得...

    Android ToolBar、TabLayout加角标显示未读消息

    当有未读消息或者新内容时,为这些组件添加角标(通常是小红点)是一种常见的提示方式。本篇将详细讲解如何在`ToolBar`和`TabLayout`上实现角标显示未读消息。 首先,我们需要了解`ToolBar`的基本使用。`ToolBar`是...

    cordova显示角标插件

    7. 配置权限:在Android平台上,可能还需要在`AndroidManifest.xml`中添加相应的权限,以允许应用修改角标。具体配置因插件版本和Android版本的不同而略有差异。 总的来说,"cordova-plugin-badge"插件是Cordova...

    Android图片右上角数字角标实现

    在Android应用开发中,我们经常需要在图片的右上角添加一个数字角标,用于显示未读消息的数量、更新提示等。这个功能在各种社交应用、通知中心等场景中非常常见。实现这一功能通常涉及到自定义View或者使用现有的库...

    Android-Android角标组件效果

    在Android应用开发中,角标(Badge)组件通常用于显示通知、未读消息计数或者状态指示等,它能够提供一种简洁的方式提醒用户有新的内容或活动需要关注。本话题将详细探讨Android中的角标组件及其效果实现。 一、角...

    底部tabbar 自定义角标

    安装这个库(通常通过CocoaPods或Carthage)后,你可以方便地为TabBarItem添加自定义角标,而无需过多关注底层实现细节。BadgeView可能提供了设置角标颜色、形状、大小和动画等功能。 具体步骤如下: - **集成...

    Android例子源码给view加红色数字提醒角标

    在Android应用开发中,给View添加红色数字提醒角标是一种常见的设计手法,它通常用于通知用户有未读消息、更新或者其他重要信息。这种设计元素在许多应用中都可以看到,如通知中心、应用图标或者菜单项。本文将深入...

    QQ消息角标显示

    1. HTML/CSS/JavaScript:在Web开发中,可以通过CSS设置样式,JavaScript动态更新角标内容,HTML结构中添加角标元素。 2. iOS开发:使用Swift或Objective-C,可以通过UIBarButtonItem或UIView自定义视图,利用通知...

    Android代码-自己做的一个android数据库复制到sdcard和一个简单的角标使用demo很简陋.zip

    在Android开发中,数据库管理是应用功能的重要组成部分...总之,这个压缩包为初学者提供了一个实践Android数据库操作和角标显示的起点,通过深入学习和理解这些代码,开发者可以进一步提升在Android应用开发中的技能。

    Android-Badge一系列的Android徽章Drawables

    在Android应用开发中,徽章(Badge)通常用于表示通知数量或者特殊状态,例如未读消息的数量。这个项目,"Android-Badge一系列的Android徽章Drawables" 提供了一种简单的方法来创建和使用徽章样式,使得开发者可以...

    Android系统通知栏适配

    3. **第三方库**:为了简化开发过程,许多第三方库如`BadgeProvider`或`Android-Badge`提供了便捷的API,使得在应用图标上添加角标变得简单。这些库通常兼容多种Android版本,减少了适配的工作量。 4. **利用...

    AndroidLabelView图片右上角的角标图片以及滚动数字效果以及圆形进度条

    在Android开发中,有时我们需要为应用的图标或者某些视图添加一些特殊的效果,例如右上角的角标图片、滚动数字以及圆形进度条等。这些元素可以用来展示通知数量、未读消息或者其他状态信息,增强用户界面的交互性和...

    Android-FuckTabLayout是修改TabLayout源码增加了角标滑动文字颜色渐变指示器填充设置等功能

    在原生TabLayout中,添加角标需要额外处理,而在"Android-FuckTabLayout"中,角标设置变得轻而易举,开发者可以通过简单的API调用来添加、修改或移除角标,使得状态提示更为直观,提高了应用的交互性。 最后,从...

    AppBadge,安卓系统.zip

    总的来说,AppBadge 是一个解决Android应用角标显示问题的实用工具,对于那些希望在应用图标上提供直观未读消息提示的开发者来说,是一个非常有价值的开源项目。通过深入理解其工作原理并将其集成到项目中,开发者...

    Android中为图标加上数字--用于未读短信数提醒

    总结起来,为Android应用图标添加未读提醒数字角标涉及到对Android通知系统的理解,以及利用ContentProvider或第三方库来实现角标的显示和清除。这个功能能显著提高用户对应用内未读通知的感知,提升应用的用户体验...

    Android应用源码之BadgeView实现在控件上显示小标签.zip

    在Android应用开发中,有时我们需要在控件上添加一个小标签,比如在应用图标右上角显示未读消息的数量,这就是BadgeView的用途。BadgeView是一个轻量级的UI组件,可以方便地展示这类提示信息。本篇文章将深入探讨...

    Android代码-BadgedView

    原作者的实现里,只能为 ImageView 添加 Badge,我看了看他的代码(开源)发现这个控件其实蛮简单的,想法挺好,但是代码写的不是特别好。而且我觉得应该能将这个控件应用到任意一个 View 上边,所以我根据我的想法...

    android 图标 消息 数量

    在`res/drawable`目录下创建一个名为`badge_layer_list.xml`的XML文件,定义图层列表,包括基础图标和角标: ```xml &lt;layer-list xmlns:android="http://schemas.android.com/apk/res/android"&gt; &lt;!-- 基础...

    Android应用源码之BadgeView实现在控件上显示小标签_实现.zip

    本篇将详细探讨如何在Android应用中实现BadgeView,以在控件上显示小标签。 首先,BadgeView的实现通常涉及到自定义视图(Custom View)的概念。开发者需要创建一个新的View类,继承自Android的View或ImageView,并...

Global site tag (gtag.js) - Google Analytics