前言
本来想研究一加手机上的天气APP,会随着每天的天气、温度不同动态的改变APP的图标是如何实现的,但始终没有想出和查阅到相关方法,毕竟是人家自己的系统APP。但是也找到了一种能动态改变图标的方法,这里做一下简单的记录。
效果图
这里点击完“改变图标”后,桌面上的APP图标不会立即改变,会有一定的延迟,需要等待Launcher刷新。
基础原理
多入口配置:activity-alias
alias是别名的意思,顾名思义,就是可以给Activity配置别名。在AndroidManifest.xml中,与基本的Activity的配置大体相似,如:
<activity
android:name=".MainActivity"
android:label="改变前的">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity-alias
android:name=".test"
android:enabled="false"
android:icon="@mipmap/icon_test"
android:label="改变后的"
android:targetActivity=".MainActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity-alias>
上面的默认的,下面的是配置的别名,也就是另一个入口。
android:name:就是注册这个组件的名字,不需要去创建test.java这个类,就是个名字,我这里的这个名称是,com.fu.changeicondemo.test
android:enabled:是否启用这个组件,也就是是否显示这个入口。
android:icon:图标
android:label:名称
android:targetActivity:这个需要注意,默认的activity没有这个属性,这个属性就是指定目标activity,与默认的activity中的name属性是一样的,需要有相应的java类文件。
然后intent-filter是一样的。
我们把别名activity的android:enabled设为true,就是开启它,桌面上就有两个入口了。
如图:
点击这两个图标效果是一样的,都是进入MainActivity。
讲到这里,相信读者应该已经明白了,如何动态改变图标了,其实就启用不同的入口就行,然后每次就只有一个入口被启用就行了,在默认情况下,我们就让启动的默认组件,其他入口组件全部禁用,需要用哪个就启用哪个,并且把其他的关闭了,避免出现多个入口,那么如何通过代码动态的设置启动或禁用呢?
如何动态设置是否启用?
那就需要packageManager,有一个这个方法:
packageManager.setComponentEnabledSetting(componentName,
PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
PackageManager.DONT_KILL_APP);
第一个参数是组件,第二个参数就是是否启用,第三个参数一般就填不杀死APP。
然后组件通过自己定义的组件名字来获取到,如
//拿到我们注册的MainActivity组件
defaultComponent = new ComponentName(getBaseContext(), "com.fu.changeicondemo.MainActivity"); //拿到默认的组件
//拿到我注册的别名test组件
testComponent = new ComponentName(getBaseContext(), "com.fu.changeicondemo.test");
到了这里,相信大家已经很清楚了如何实现了,下面附上代码。
代码
MainActivity.java
public class MainActivity extends AppCompatActivity {
private ComponentName defaultComponent;
private ComponentName testComponent;
private PackageManager packageManager;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//拿到当前activity注册的组件名称
ComponentName componentName = getComponentName();
//拿到我们注册的MainActivity组件
defaultComponent = new ComponentName(getBaseContext(), "com.fu.changeicondemo.MainActivity"); //拿到默认的组件
//拿到我注册的别名test组件
testComponent = new ComponentName(getBaseContext(), "com.fu.changeicondemo.test");
packageManager = getApplicationContext().getPackageManager();
}
public void changeIcon(View view) {
disableComponent(defaultComponent);
enableComponent(testComponent);
}
public void changeDefaultIcon(View view) {
enableComponent(defaultComponent);
disableComponent(testComponent);
}
/**
* 启用组件
*
* @param componentName
*/
private void enableComponent(ComponentName componentName) {
int state = packageManager.getComponentEnabledSetting(componentName);
if (state == PackageManager.COMPONENT_ENABLED_STATE_ENABLED) {
//已经启用
return;
}
packageManager.setComponentEnabledSetting(componentName,
PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
PackageManager.DONT_KILL_APP);
}
/**
* 禁用组件
*
* @param componentName
*/
private void disableComponent(ComponentName componentName) {
int state = packageManager.getComponentEnabledSetting(componentName);
if (state == PackageManager.COMPONENT_ENABLED_STATE_DISABLED) {
//已经禁用
return;
}
packageManager.setComponentEnabledSetting(componentName,
PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
PackageManager.DONT_KILL_APP);
}
}
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="com.fu.changeicondemo.MainActivity">
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="改变图标"
android:onClick="changeIcon"/>
<Button
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="改回默认图标"
android:onClick="changeDefaultIcon" />
</LinearLayout>
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.fu.changeicondemo">
<application
android:allowBackup="true"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name"
android:roundIcon="@mipmap/ic_launcher_round"
android:supportsRtl="true"
android:theme="@style/AppTheme">
<activity
android:name=".MainActivity"
android:label="改变前的">
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
<activity-alias
android:name=".test"
android:enabled="false"
android:icon="@mipmap/icon_test"
android:label="改变后的"
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>
总结
这个方法能动态改变图标和名称、但是需要提前把“坑位”站住,需要用哪个就用哪个。还有一个问题就是,在调用完相应的改变图标的方法时候需要等待Launcher刷新,而在这个时间里,在去点击图标就会提示“未安装该应用”。如图: