Android:动态更新app启动图标和应用名

一、需求背景

每逢重要佳节,很多应用启动图标会自动更新为对应佳节的图标,应用无需更新。

二、效果图

更新后的启动图标和应用名称

三、实现流程

Android app只能替换内置的icon,因此需要提前将logo图标放入App资源文件件里

实际项目App更新桌面启动图标由服务器端控制,可以在App启动页里请求全局配置接口,根据接口返回的是否更新启动图标字段值进行处理。
大多数都是用activity-alias方式更新启动图标和应用名称,但是谷歌建议新建activity继承启动页的activity,新建的activity里面是空的。

下面的代码采用新建activity继承启动页的activity方式实现动态更新App启动图标和应用名称

3.1 AndroidManifest.xml代码
bash 复制代码
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools">

    <application
        android:allowBackup="true"
        android:dataExtractionRules="@xml/data_extraction_rules"
        android:fullBackupContent="@xml/backup_rules"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/Theme.Test"
        tools:targetApi="31">
        <activity
            android:name=".ui.activity.LauncherActivity"
            android:exported="true">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <!--   更改启动图标和应用名称     -->
        <!--        <activity-alias-->
        <!--            android:name=".ui.activity.NewMainActivity"-->
        <!--            android:enabled="false"-->
        <!--            android:exported="true"-->
        <!--            android:icon="@mipmap/ic_launcher_shlx"-->
        <!--            android:label="@string/app_name2"-->
        <!--            android:targetActivity=".ui.activity.MainActivity">-->
        <!--            <intent-filter>-->
        <!--                <action android:name="android.intent.action.MAIN" />-->

        <!--                <category android:name="android.intent.category.LAUNCHER" />-->
        <!--            </intent-filter>-->

        <!--        </activity-alias>-->

        <activity
            android:name=".ui.activity.LauncherActivityNew"
            android:enabled="false"
            android:exported="true"
            android:icon="@mipmap/ic_launcher_mid_autumn_festival"
            android:label="@string/app_name2"
            android:targetActivity=".ui.activity.MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

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

        <activity
            android:name=".ui.activity.MainActivity"
            android:launchMode="standard"
            android:screenOrientation="portrait" />

        <activity
            android:name=".ui.activity.SettingActivity"
            android:launchMode="standard"
            android:screenOrientation="portrait" />
    </application>

</manifest>

注意:

3.2 activity_launcher.xml代码
bash 复制代码
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".ui.activity.LauncherActivity">

    <TextView
        android:id="@+id/tvSetLogo"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:layout_margin="16dp"
        android:background="@color/black"
        android:gravity="center"
        android:text="启动页"
        android:textColor="@color/white"
        android:textSize="20sp"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

</androidx.constraintlayout.widget.ConstraintLayout>
3.3 LauncherActivity.kt代码
bash 复制代码
package com.example.test.ui.activity

import android.content.ComponentName
import android.content.Intent
import android.content.pm.PackageManager
import android.os.Bundle
import androidx.appcompat.app.AppCompatActivity
import com.example.test.R
import com.example.test.utils.LogUtil


open class LauncherActivity : AppCompatActivity() {
    private lateinit var mPackageManager: PackageManager
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
//        enableEdgeToEdge()
        setContentView(R.layout.activity_launcher)
//        ViewCompat.setOnApplyWindowInsetsListener(findViewById(R.id.main)) { v, insets ->
//            val systemBars = insets.getInsets(WindowInsetsCompat.Type.systemBars())
//            v.setPadding(systemBars.left, systemBars.top, systemBars.right, systemBars.bottom)
//            insets
//        }

//        LogUtil.i("获取到的路径 simpleName:${ LunchActivity::class.java.simpleName}")
//        LogUtil.i("获取到的路径 packageName:${ LunchActivity::class.java.`package`.name}")
//        LogUtil.i("获取到的路径 name:${ LunchActivity::class.java.name}")
        /*
            获取到的路径 simpleName:LunchActivity
            获取到的路径 packageName:com.example.test.ui.activity
            获取到的路径 name:com.example.test.ui.activity.LunchActivity
         */


        initData()
        initView()
        initEvent()

        if (this::class.java.simpleName.equals(LauncherActivityNew::class.java.simpleName)) {
            LogUtil.i("没有 执行 updateLauncherIcon")
            startActivity(Intent(this, MainActivity::class.java))
            finish()
        } else {
            updateLauncherIcon()
        }
    }

    private fun initData() {
        mPackageManager = applicationContext.packageManager
    }

    private fun initView() {

    }

    private fun initEvent() {

    }

    private fun updateLauncherIcon() {
        LogUtil.i("执行了 updateLauncherIcon")
//            var festervalIcon = ComponentName(baseContext, "com.example.test.ui.activity.LunchActivity")
        var festervalIcon = ComponentName(baseContext, LauncherActivityNew::class.java.name)
        enableComponent(festervalIcon)
        disableComponent(componentName)
        startMainActivity()
    }

    /*
        PackageManager.DONT_KILL_APP
        PackageManager.SYNCHRONOUS
     */

    //启用组件
    private fun enableComponent(componentName: ComponentName) {
        mPackageManager.setComponentEnabledSetting(
            componentName,
            PackageManager.COMPONENT_ENABLED_STATE_ENABLED, PackageManager.DONT_KILL_APP
        )
    }

    //隐藏组件
    private fun disableComponent(componentName: ComponentName) {
        mPackageManager.setComponentEnabledSetting(
            componentName,
            PackageManager.COMPONENT_ENABLED_STATE_DISABLED, PackageManager.DONT_KILL_APP
        )
    }

    private fun startMainActivity() {
        val intent = Intent(this, MainActivity::class.java)
        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_DOCUMENT or Intent.FLAG_ACTIVITY_MULTIPLE_TASK)
        this.startActivity(intent)
        finish()
    }
}
3.4 LauncherActivityNew.kt代码
bash 复制代码
class LauncherActivityNew : LauncherActivity() {
}
3.5 activity_main.xml代码
bash 复制代码
<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".ui.activity.LauncherActivity">

    <TextView
        android:id="@+id/tvSetLogo"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:layout_margin="16dp"
        android:background="@color/black"
        android:gravity="center"
        android:text="打开设置页面"
        android:textColor="@color/white"
        android:textSize="20sp"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />


</androidx.constraintlayout.widget.ConstraintLayout>
3.6 MainActivity.kt代码
bash 复制代码
package com.example.test.ui.activity

import android.content.Intent
import android.os.Bundle
import android.widget.TextView
import androidx.appcompat.app.AppCompatActivity
import com.example.test.R

class MainActivity : AppCompatActivity(R.layout.activity_main) {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        findViewById<TextView>(R.id.tvSetLogo).setOnClickListener {
            startActivity(
                Intent(
                    this,
                    SettingActivity::class.java
                )
            )
        }
    }
}
3.7 activity_setting.xml代码
bash 复制代码
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

    <TextView
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:layout_gravity="center"
        android:gravity="center"
        android:text="设置"
        android:textColor="@color/black"
        android:textSize="20sp" />

</LinearLayout>
3.8 SettingActivity.kt代码
bash 复制代码
package com.example.test.ui.activity

import android.os.Bundle
import androidx.appcompat.app.AppCompatActivity
import com.example.test.R

class SettingActivity: AppCompatActivity(R.layout.activity_setting) {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
    }
}
相关推荐
OkeyProxy5 分钟前
設置Android設備全局代理
android·代理模式·proxy模式·代理服务器·海外ip代理
刘志辉40 分钟前
vue传参方法
android·vue.js·flutter
前期后期3 小时前
Android OkHttp源码分析(一):为什么OkHttp的请求速度很快?为什么可以高扩展?为什么可以高并发
android·okhttp
轻口味6 小时前
Android应用性能优化
android
全职计算机毕业设计6 小时前
基于 UniApp 平台的学生闲置物品售卖小程序设计与实现
android·uni-app
dgiij6 小时前
AutoX.js向后端传输二进制数据
android·javascript·websocket·node.js·自动化
SevenUUp7 小时前
Android Manifest权限清单
android
高林雨露7 小时前
Android 检测图片抓拍, 聚焦图片后自动完成拍照,未对准图片的提示请将摄像头对准要拍照的图片
android·拍照抓拍
wilanzai7 小时前
Android View 的绘制流程
android
INSBUG8 小时前
CVE-2024-21096:MySQLDump提权漏洞分析
android·adb