双11快到了,不给你的APP加上自动换图标的功能吗?

注:此方案在部分机型存在不兼容现象,具体表现不一致,可参考文章评论的反馈。如果想投入生产,请务必先评估出现的风险点能不能接受。

前言

也许你也注意到了,在临近双11之际,手机上电商类APP的应用图标已经悄无声息换成了双11专属图标,比如某宝和某东:


8efea43731cdf2579eb4fb031c774737.jpg

可能你会说,这有什么奇怪的,应用市场开启自动更新不就可以了么?

真的是这样吗?

为此,我特意查看了我手机上的某宝APP的当前版本,并对比了历史版本上的图标,发现并不对应。

281048112ddef4486bc48c6abac63bb5.jpg
41bdaa904abf3655fb81193789d5e3cc.jpg

默认是88会员节专属图标,而现在显示的是双11图标。

那么,作为开发者的嗅觉,让你自然而然想要从技术角度揣测是怎么实现的,而这便是这篇文章想要与你分享的。

知识储备

<activity-alias>

某一个Activity 的别名,用于实例化该目标Activity。目标必须与别名在同一应用中,并且在清单中必须在别名之前进行声明。
介绍下几个重要的属性:
android:enabled:必须设为“true”,系统才能通过别名实例化目标 Activity
android:icon:通过别名呈现给用户时目标 Activity 的图标。
android:name:别名的唯一名称。与目标 Activity 的名称不同,别名名称是任意的,它不引用实际类。
android:targetActivity:可通过别名激活的 Activity 的名称。

PackageManager#setComponentEnabledSetting

可以利用 PackageManager 在清单文件中所定义的任何组件上切换启用状态,包括您想启用或停用的任何一个Activity。

有了以上知识储备后,下面就该剖析一下这个需求的具体场景了。

场景剖析

以电商类APP双11活动为例,在双11活动开始前的某个时间点(比如10天前)就要开始对活动的预热,此时就要实现图标的自动更换,而在活动结束之后,也必须要能更换回正常图标,并且要求过程尽量对用户无感知,更不能影响用户对APP的正常使用。

具体拆分成要实现的功能点便是:图标更换、自动操作、用户无感知。

方案实现

1.图标更换:禁用Launcher组件,启用Alias组件,并将targetActivity指向原先的Launcher组件。
2.自动操作:指定日期转换为时间戳,并与当前时间戳对比,超过预设时间则执行替换操作。
3.用户无感知:尽量选择APP不活跃的阶段的,比如切换应用/回到桌面时。

代码实践

首先,我们需要在AndroidManifest清单文件中添加<activity-alias>元素,默认为禁用状态,name属性作为我们找到此组件的唯一标志,而icon属性即是我们要替换的图标资源,并通过targetActivity属性将作为LANCHUER的SplashActivity作为实例化的目标 Activity:

<activity android:name=".SplashActivity">
    <intent-filter>
        <action android:name="android.intent.action.MAIN" />
        <category android:name="android.intent.category.LAUNCHER" />
    </intent-filter>
</activity>

<!--88会员节专属Activity别名-->
<activity-alias
    android:name=".SplashAliasActivity"
    android:enabled="false"
    android:icon="@mipmap/ic_launcher_88"
    android:targetActivity=".SplashActivity">
    <intent-filter>
        <action android:name="android.intent.action.MAIN" />
        <category android:name="android.intent.category.LAUNCHER" />
    </intent-filter>
</activity-alias>

<!--双11专属Activity别名-->
<activity-alias
    android:name=".SplashAlias2Activity"
    android:enabled="false"
    android:icon="@mipmap/ic_launcher_11_11"
    android:targetActivity=".SplashActivity">
    <intent-filter>
        <action android:name="android.intent.action.MAIN" />
        <category android:name="android.intent.category.LAUNCHER" />
    </intent-filter>
</activity-alias>

随后,我们图标替换的工作视作一项任务,定义一个数据类:

/**
 * 切换图标任务
 */
data class SwitchIconTask (val launcherComponentClassName: String,  // 启动器组件类名
                           val aliasComponentClassName: String,  // 别名组件类名
                           val presetTime: Long,            // 预设时间
                           val outDateTime: Long)           // 过期时间

定义一个LauncherIconManager单例,负责图标更换相关的工作。开放添加图标切换任务的接口,做好参数合法性的校验:

/**
 * 启动器图标管理器
 */
object LauncherIconManager {

    /** 切换图标任务Map */
    private val taskMap: LinkedHashMap<String, SwitchIconTask> = LinkedHashMap()

    /**
     * 添加图标切换任务
     * @param newTasks 新任务,可以传多个
     */
    fun addNewTask(vararg newTasks: SwitchIconTask) {
        for (newTask in newTasks) {
            // 防止重复添加任务
            if (taskMap.containsKey(newTask.aliasComponentClassName)) return
    
            // 校验任务的预设时间和过期时间
            for (queuedTask in taskMap.values) {
                if (newTask.presetTime > newTask.outDateTime) throw IllegalArgumentException("非法的任务预设时间${newTask.presetTime}, 不能晚于过期时间")
                if (newTask.presetTime <= queuedTask.outDateTime) throw IllegalArgumentException("非法的任务预设时间${newTask.presetTime}, 不能早于已添加任务的过期时间")
            }
    
            taskMap[newTask.aliasComponentClassName] = newTask
        }
    }
    
    ...
}
LauncherIconManager.addNewTask(
    SwitchIconTask(
        SplashActivity::class.java.name,
        "$packageName.SplashAliasActivity",
        format.parse("2020-08-02").time,
        format.parse("2020-08-09").time
    ),
    SwitchIconTask(
        SplashActivity::class.java.name,
        "$packageName.SplashAlias2Activity",
        format.parse("2020-11-05").time,
        format.parse("2020-11-12").time
    )
)

通过Application#registerActivityLifecycleCallbacks方法注册了对应用内Activity生命周期的监听,通过是否有活跃状态的Activity判断应用是否进入了后台:

/**
 * 应用运行状态注册器
 */
object RunningStateRegister {

    fun register(application: Application, callback: StateCallback) {
        application.registerActivityLifecycleCallbacks(object : SimpleActivityLifecycleCallbacks() {
            private var startedActivityCount = 0
            override fun onActivityStarted(activity: Activity) {
                if (startedActivityCount == 0) {
                    callback.onForeground()
                }
                startedActivityCount++
            }

            override fun onActivityStopped(activity: Activity) {
                startedActivityCount--
                if (startedActivityCount == 0) {
                    callback.onBackground()
                }
            }
        })
    }
    
}   
class BaseApplication : Application() {

    override fun onCreate() {
        super.onCreate()
        LauncherIconManager.register(this)
    }
}

判断应用进入后台后,就可以开始对图标的更换工作了:

/**
 * 启动器图标管理器
 */
object LauncherIconManager {
    ...
    
    /**
     * 注册以监听应用运行状态
     */
    fun register(application: Application) {
        RunningStateRegister.register(application, object: RunningStateRegister.StateCallback{
            override fun onForeground() {
            }
    
            override fun onBackground() {
                proofreadingInOrder(application)
            }
        })
    }
    
    /**
     * 依次校对预设时间
     * @param context 上下文
     */
    fun proofreadingInOrder(context: Context) {
        for (task in taskMap.values) {
            if (proofreading(context, task)) break
        }
    }
    
    /**
     * 校对预设时间/过期时间
     * @param context 上下文
     * @return true 已过预设时间      false 未达预设时间或已过期
     */
    private fun proofreading(context: Context, task: SwitchIconTask) =
        when {
            isPassedOutDateTime(task) -> {
                disableComponent(context, ActivityUtil.getLauncherActivityName(context)!!)
                enableComponent(context, task.launcherComponentClassName)
                false
            }
            isPassedPresetTime(task) -> {
                disableComponent(context, ActivityUtil.getLauncherActivityName(context)!!)
                enableComponent(context, task.aliasComponentClassName)
                true
            }
            else -> false
        }
    
    /**
     * 是否已超过预设时间
     * @param task 任务
     */
    private fun isPassedPresetTime(task: SwitchIconTask) =
        System.currentTimeMillis() > task.presetTime
    
    /**
     * 是否已超过过期时间
     * @param task 任务
     *
     */
    private fun isPassedOutDateTime(task: SwitchIconTask) =
        System.currentTimeMillis() > task.outDateTime

    ...
}        

以上代码均已上传到GitHub。核心的类都封装到Library模块了,并提供Demo模块演示如何使用。
如果觉得项目不错的话点个Star吧~
https://github.com/madchan/LauncherIconLib

效果预览

1604935350118.gif

总结

通过以上构建的方案,便可让我们的APP在预设的时间点实现对应用图标的自动替换,缺点是只能加载随APK打包的图片资源,适用于运营活动时间相对固定的的场景。

参考文章

<activity-alias>
https://developer.android.google.cn/guide/topics/manifest/activity-alias-element

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,839评论 6 482
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,543评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 153,116评论 0 344
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,371评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,384评论 5 374
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,111评论 1 285
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,416评论 3 400
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,053评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,558评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,007评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,117评论 1 334
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,756评论 4 324
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,324评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,315评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,539评论 1 262
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,578评论 2 355
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,877评论 2 345

推荐阅读更多精彩内容