转载请注明出处: http://blog.youkuaiyun.com/xiaohao0724/article/details/62424427
每年到11.11或5.18时候我们会发现很多电商App的icon在我们没有升级App的情况下却变了,今天就来带大家来实现这个效果。先上图如下: 
要实现这个功能需要用到定义在清单文件中的activity-alias标签-Activity别名标签。
activity-alias作为一个已存在Activity的别名,则应该可以通过该别名标签声明快速打开目标Activity。因此activity-alias可用来设置某个Activity的快捷入口,可以放在桌面上或者通过该别名被其他组件快速调起。该标签元素支持一些属性及intent-filter、meta-data等配置,因此可以触发一些跟目标Activity不同的功能逻辑,虽然打开的是同一个Activity。举个简单的例子,如之前需要先打开主界面,然后才能点击进入某个Activity,如果使用activity-alias为该Activity配置一个快捷入口,甚至可以为其在桌面生成一个图标,然后点击桌面图标可直接进入该Activity,该功能可满足某些需要快速到达功能界面的需求。
activity-alias的属性是activity属性的子集,如果是activity-alias和activity共有的属性,则以activity-alias为准,目标Activity中的设置并不会在activity-alias中生效;如果是仅activity才有的属性,则为目标Activity配置的属性会在activity-alias中生效。
注意:activity-alias标签元素必须声明在目标Acitvity对应的activity标签元素之后,否则会编译错误。
activity-alias语法
<activity-alias android:enabled=["true" | "false"]
android:exported=["true" | "false"]
android:icon="drawable resource"
android:label="string resource"
android:name="string"
android:permission="string"
android:targetActivity="string" >
</activity-alias>
android:enable
该属性用来决定目标Activity可否通过别名被系统实例化,默认为true。需要注意的是application也有enable属性,只用当它们同时为true时,activity-alias的enable才生效。
android:exported
该属性为true的话,则目标Activity可被其他应用调起,如为false则只能被应用自身调起。其默认值根据activity-alias是否包含intent-filter元素决定,如果有的话,则默认为true;没有的话则为false。其实也很好理解,如果有intent-filter,则目标Activity可以匹配隐式Intent,因此可被外部应用唤起;如果没有intent-filter,则目标Activity要被调起的话必须知道其精确类名,因为只有应用本身才知道精确类名,所以此时默认为false。
android:icon
该属性就比较好玩了,允许自定义icon,可以不同于应用本身在桌面的icon。如果需要在桌面上创建快捷入口,也许产品会要求换个不同的icon。
android:label
该属性类似于android:icon,图标都换了,换个名称也合情合理吧,此属性就是为此而生的。
android:name
该属性可以为任意字符串,但最好符合类名命名规范。activity元素的name属性实质上都会指向一个具体的Activity类,而activity-alias的name属性仅作为一个唯一标识而已。
android:permission
该属性指明了通过别名声明调起目标Activity所必需的权限。
android:targetActivity
该属性指定了目标Activity,即通过activity-alias调起的Activity是哪个,此属性其实类似于activity标签中的name属性,需要规范的Activity包名类名。
下面我们来用activity-alias实现动态更换桌面Icon图标。
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name=".MainActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
</intent-filter>
</activity>
<activity-alias
android:name="main.normal"
android:enabled="true"
android:icon="@drawable/ic_launcher"
android:label="LauncherDemo1"
android:targetActivity=".MainActivity" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity-alias>
<activity-alias
android:name="main.event"
android:enabled="false"
android:icon="@drawable/icon"
android:label="LauncherDemo2"
android:targetActivity=".MainActivity" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity-alias>
</application>
定义一个工具类来实现对Icon的设置
ublic class IconUtil {
public static void setIcon(Context context, String enabledActivity) {
PackageManager packageManager = context.getPackageManager();
// 根据传入的enabledActivity判断是让ACTIVITY_ALIAS_1失效或生效
packageManager
.setComponentEnabledSetting(
new ComponentName(context,
Constants.ACTIVITY_ALIAS_Main1),
Constants.ACTIVITY_ALIAS_Main1.equals(enabledActivity) ? PackageManager.COMPONENT_ENABLED_STATE_ENABLED
: PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
PackageManager.DONT_KILL_APP);
// 根据传入的enabledActivity判断是让ACTIVITY_ALIAS_Main2失效或生效
packageManager
.setComponentEnabledSetting(
new ComponentName(context,
Constants.ACTIVITY_ALIAS_Main2),
Constants.ACTIVITY_ALIAS_Main2.equals(enabledActivity) ? PackageManager.COMPONENT_ENABLED_STATE_ENABLED
: PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
PackageManager.DONT_KILL_APP);
// kill掉桌面app,让它自动重启。以便观察到动态更换后的Icon
ActivityManager am = (ActivityManager) context
.getSystemService(Activity.ACTIVITY_SERVICE);
Intent i = new Intent(Intent.ACTION_MAIN);
i.addCategory(Intent.CATEGORY_HOME);
i.addCategory(Intent.CATEGORY_DEFAULT);
List<ResolveInfo> resolves = packageManager.queryIntentActivities(i, 0);
for (ResolveInfo res : resolves) {
if (res.activityInfo != null) {
am.killBackgroundProcesses(res.activityInfo.packageName);
}
}
}
}
<uses-permission android:name="android.permission.KILL_BACKGROUND_PROCESSES" />