Android动态更换APP图标及名称

前言

本来想研究一加手机上的天气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刷新,而在这个时间里,在去点击图标就会提示“未安装该应用”。如图:
这里写图片描述

评论 25
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值