转载请注明原创出处,谢谢!
- GitHub: @Ricco
ps:很早以前,在写日历需求的时候,特意留意了一下小米日历app的ui,发现了一个有意思的交互(月和周之间的切换),随着小米日历app的更新,发现了一个更有意思的地方,app的icon是随着日期,天气,温度改变的。联想到马上要到双11了,手机里的很多app的icon改成了双11版。所以简单的了解了一下
本文实现的效果绝对不是小米日历的效果,差别还是很大的。
所以本文章为伪动态修改Icon。
实现还是很简单的,主要用到的是activity-alias这个便签,也就是activity的一个别名,android允许一个activity有多个别名。
在写代码前,先准备几个icon,我装备了2个,加上原生的,一共3个icon。
先看AndroidManifest.xml代码
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.demo.icon">
<uses-permission android:name="android.permission.KILL_BACKGROUND_PROCESSES" />
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity android:name=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<!--创建多个不同的入口-->
<activity-alias
android:name=".MainActivity_Test1"
android:enabled="false"
android:icon="@mipmap/ic_launcher1"
android:label="Test1"
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=".MainActivity_Test2"
android:enabled="false"
android:icon="@mipmap/ic_launcher2"
android:label="Test2"
android:targetActivity=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity-alias>
</application>
</manifest>
在AndroidManifest.xml中设置了2个activity-alias,其中MainActivity_Test1的icon是ic_launcher1,MainActivity_Test2的icon是ic_launcher2。
android:enabled="false"代码决定了是否起作用,如果想要有一个有效,就配置成 android:enabled="true"。
因为我的代码是默认使用MainActivity的icon。所以2个activity-alias设置为fale。如果你有兴趣,将他们都设置成true,运行成功后,你会神奇的发现桌面上除了MainActivity的icon,还多了2个activity-alias的icon。
xml写好后,开始写MainActivity代码。
public class MainActivity extends AppCompatActivity {
private static final String TAG = "MainActivity";
private PackageManager packageManager;
private ActivityManager activityManager;
private ComponentName componentNameDefault;
private ComponentName componentName1;
private ComponentName componentName2;
private static final String DEFAULT = "com.demo.icon.MainActivity";
private static final String ACTIVITY_ALIAS_1 = "com.demo.icon.MainActivity_Test1";
private static final String ACTIVITY_ALIAS_2 = "com.demo.icon.MainActivity_Test2";
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
// componentNameDefault = getComponentName(); // getComponentName得到的是当前的,不是MainActivity
Log.i(TAG, "onCreate:------> " + getComponentName());
packageManager = getPackageManager();
activityManager = (ActivityManager) getSystemService(Activity.ACTIVITY_SERVICE);
componentNameDefault = new ComponentName(this, DEFAULT);
componentName1 = new ComponentName(this, ACTIVITY_ALIAS_1);
componentName2 = new ComponentName(this, ACTIVITY_ALIAS_2);
// 显示默认
findViewById(R.id.Default).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
disableComponent(componentName1);
disableComponent(componentName2);
enableComponent(componentNameDefault);
start();
}
});
// 显示1
findViewById(R.id.Test1).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
disableComponent(componentNameDefault);
disableComponent(componentName2);
enableComponent(componentName1);
start();
}
});
// 显示2
findViewById(R.id.Test2).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
disableComponent(componentNameDefault);
disableComponent(componentName1);
enableComponent(componentName2);
start();
}
});
}
/**
* 立即开始执行,如果不执行start方法,根据ROM的不同,在禁用了组件之后,会等一会,Launcher也会自动刷新图标。
*/
private void start() {
Intent intent = new Intent(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_HOME);
intent.addCategory(Intent.CATEGORY_DEFAULT);
List<ResolveInfo> resolves = packageManager.queryIntentActivities(intent, PackageManager.COMPONENT_ENABLED_STATE_DEFAULT); // 默认启用状态
for (ResolveInfo res : resolves) {
if (res.activityInfo != null) {
activityManager.killBackgroundProcesses(res.activityInfo.packageName); // 杀死后台进程
}
}
}
private void enableComponent(ComponentName componentName) { // 显示
packageManager.setComponentEnabledSetting(componentName, PackageManager.COMPONENT_ENABLED_STATE_ENABLED, PackageManager.DONT_KILL_APP);
}
private void disableComponent(ComponentName componentName) { // 隐藏
packageManager.setComponentEnabledSetting(componentName, PackageManager.COMPONENT_ENABLED_STATE_DISABLED, PackageManager.DONT_KILL_APP);
}
}
MainActivity有3个按钮,点击不同按钮,就可以在家吗中看到切换成功的icon。
注意
如果你在显示和隐藏后,不执行start()方法,icon的图片也可以被换掉,只是根据ROM的不同,会等一会自动刷新图标。
不要使用Activity的getComponentName()方法,得到默认的ComponentName,getComponentName得到的是当前的ComponentName,不是“com.demo.icon.MainActivity”。
- activity-alias最好不要删除。
如果我们在应用v1.0的版本在某些特定的情况下使用的activity-alias1,而在应用v2.0的版本中删除了activity-alias1的代码,那么你的应用,将在桌面上找不到,也打不开。这种魔鬼操作对于公司是绝对不可以的。
执行start方法,当年退出app的时候,桌面会进入一个空白期(我不知道该桌面描述,就是类似桌面重新加载一样),这就是我为什么所这是伪动态修改Icon。
icon文件是写死的,如果你想从网络上获取,很抱歉,我还不知道怎么实现。