前言
動態更換App圖標,網上可以收搜到很多,這里也是參考前人經驗,讀完本文可以得到,如何動態更換桌標(非網絡獲取桌標圖片),標志位的闡述,更加透徹的理解.
用到的知識
activity-alias并不是代表一個Activity,而是代表一個已經存在的Activity的別名。
它使用在清單文件中,類似Activity標簽。它可用來設置某個Activity的快捷入口
activity-alias基本用法
android:exported=["true" | "false"]
android:icon="drawable resource"
android:label="string resource"
android:name="string"
android:permission="string"
android:targetActivity="string" >
...
屬性解釋:
屬性
含義
enabled
是否生效。配置多個activity-alias時,如果只想一個生效,就設置一個為true
exported
是否可以被其他應用調起,配置intent-filter時默認為true,未配置intent-filter時默認為false,只能被應用自身調起
icon
自定義生效時的icon
label
作用同Activity標簽中的label屬性,主要表現為桌面上的app名稱和activity的title的名稱
name
該activity-alias的名字
permission
指明通過別名聲明調起目標Activity所必需的權限
targetActivity
指明目標Activity,類似于Activity標簽中的name屬性,需寫明包類路徑。表明通過activity-alias調起的是哪個Activity
使用
首先配置AndroidManifest.xml,設置別名
android:name=".activitys.WelcomeActivity"
android:configChanges="keyboard|keyboardHidden|orientation"
android:hardwareAccelerated="true"
android:label="@string/app_name"
android:screenOrientation="portrait"
android:theme="@style/Acrivity_Fullscreen">
android:name=".changeLauncherIconActivity"
android:configChanges="keyboard|keyboardHidden|orientation"
android:enabled="false"
android:icon="@drawable/yishijie_logo"
android:label="@string/app_name"
android:screenOrientation="portrait"
android:targetActivity=".activitys.WelcomeActivity">
這里要注意的是
android:name 標識:主要用于在代碼中獲取此組件enable的狀態;
android:targetActivity標識,targetActivity標識就是點擊后跳轉的Activity;
icon和lable分別是啟動圖標和桌面名稱
代碼配置
思路是首先獲取服務端下發接口,緩存到本地,等用戶退出主頁的時候執行更換圖標的邏輯
獲取服務端接口,接口提示更換節日圖標
判斷要顯示組件的狀態是否為顯示狀態COMPONENT_ENABLED_STATE_ENABLED
private boolean isComponentState(ComponentName componentName) {
return mPackageManager.getComponentEnabledSetting(componentName) != PackageManager.COMPONENT_ENABLED_STATE_ENABLED;
}
注意:這里的標示位
屬性
含義
COMPONENT_ENABLED_STATE_DEFAULT
默認狀態,xml預設的狀態
COMPONENT_ENABLED_STATE_ENABLED
此組件或應用程序已明確啟用,無論其清單中指定了什么。
COMPONENT_ENABLED_STATE_DISABLED
此組件或應用程序已明確禁用,無論其清單中指定了什么。
COMPONENT_ENABLED_STATE_DISABLED_USER
用戶已明確禁用該應用程序,無論其在清單中指定了什么。
COMPONENT_ENABLED_STATE_DISABLED_UNTIL_USED
his application should be considered, until the point where the user actually wants to use it. (這個不清楚怎么翻才好,沒有使用過)
如果不是則設置其可見,否則不變
注意這里設置標志位是永久性的,即使App升級獲取此組件狀態時,也是之前的值
private void enableComponent(ComponentName componentName) {
//此方法用以啟用和禁用組件,會覆蓋Androidmanifest文件下定義的屬性
mPackageManager.setComponentEnabledSetting(componentName,
PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
PackageManager.DONT_KILL_APP);
}
完整代碼
public class ChangeAppIconUtils {
private PackageManager mPackageManager;
//默認桌標
private static final String DEFAULT_ICON = "com.x.x.activitys.WelcomeActivity";
//活動桌標
private static final String ANTHER_ICON = "com.x.x.changeLauncherIconActivity";
//緩存文件鍵值
public static final String KEY_LAUNCHER_ICON = "key_launcher_icon";
public ChangeAppIconUtils(PackageManager mPackageManager) {
this.mPackageManager = mPackageManager;
}
/**
* 啟動組件
*
* @param componentName 組件名
*/
private void enableComponent(ComponentName componentName) {
//此方法用以啟用和禁用組件,會覆蓋Androidmanifest文件下定義的屬性
mPackageManager.setComponentEnabledSetting(componentName,
PackageManager.COMPONENT_ENABLED_STATE_ENABLED,
PackageManager.DONT_KILL_APP);
}
/**
* 禁用組件
*
* @param componentName 組件名
*/
private void disableComponent(ComponentName componentName) {
mPackageManager.setComponentEnabledSetting(componentName,
PackageManager.COMPONENT_ENABLED_STATE_DISABLED,
PackageManager.DONT_KILL_APP);
}
/**
* 當前組件的狀態,判斷當前enable狀態
* 即使xml里面設置enable=false 標志位第一次獲取時 還是COMPONENT_ENABLED_STATE_DEFAULT
* 所以這里判斷是否為enable
*
* @param componentName return true 未被應用為可顯示
*/
private boolean isComponentState(ComponentName componentName) {
//默認圖標且為默認狀態則返回false
return !(DEFAULT_ICON.equals(componentName.getClassName()) && mPackageManager.getComponentEnabledSetting(componentName) == PackageManager.COMPONENT_ENABLED_STATE_DEFAULT)
&& mPackageManager.getComponentEnabledSetting(componentName) != PackageManager.COMPONENT_ENABLED_STATE_ENABLED;
}
/**
* 更換app 圖標
*
* @param context context
* @param changeIcon changeIcon
*/
private void changeIconState(Context context, String changeIcon) {
ComponentName defaultIcon = new ComponentName(context, DEFAULT_ICON);
ComponentName otherIcon = new ComponentName(context, ANTHER_ICON);
//判斷狀態
if (DEFAULT_ICON.equals(changeIcon)) {//設置默認icon
boolean componentState = isComponentState(defaultIcon);
if (componentState) {//如果不一樣則設置
enableComponent(defaultIcon);
disableComponent(otherIcon);
// restartSystemLauncher(context, mPackageManager);
}
} else {//其它icon
boolean componentState = isComponentState(otherIcon);
if (componentState) {
enableComponent(otherIcon);
disableComponent(defaultIcon);
// restartSystemLauncher(context, mPackageManager);
}
}
}
/**
* 沒啥用,有的rom不會讓你殺掉Launcher進程,例如華為,VIVO
* @param context
* @param pm
*/
private void restartSystemLauncher(Context context, PackageManager pm) {
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 resolves = pm.queryIntentActivities(i, 0);
for (ResolveInfo res : resolves) {
if (res.activityInfo != null && am != null) {
am.killBackgroundProcesses(res.activityInfo.packageName);
}
}
}
public void setAppLauncherIcon(Context context, String tagName) {
if (!TextUtils.isEmpty(tagName)) {
if ("icon2".equals(tagName)) {
changeIconState(context, ANTHER_ICON);
} else {
changeIconState(context, DEFAULT_ICON);
}
}
}
}
使用的時候只需要
new ChangeAppIconUtils(getPackageManager()).setAppLauncherIcon(getApplicationContext(), sharePreUtils.getStringValue(ChangeAppIconUtils.KEY_LAUNCHER_ICON, ""));
這里的sharepreUtils是工具類獲取接口中下發的狀態值,icon1默認圖標,icon2為節日圖標,因為之前有人說會導致app的重啟,所以這里的操作時放在主Activity onDestory里面執行的.
問題
目前已知的問題,當改完圖標之后,使用AS再次啟動會無法啟動,把快速啟動關掉就可以了
Error while executing: am start -n "in.myinnos.changeappiconandname/in.myinnos.changeappiconandname.MainActivity-settings" -a android.intent.action.MAIN -c android.intent.category.LAUNCHER
Starting: Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] cmp=in.myinnos.changeappiconandname/.MainActivity-settings }
Error type 3
Error: Activity class {in.myinnos.changeappiconandname/in.myinnos.changeappiconandname.MainActivity-settings} does not exist.
Error while Launching activity
這里我使用打包的方式覆蓋安裝沒有出現這個問題
改過圖標后,會過一會兒圖標才會改變,有的(華為)改變之前點擊會提示,但是桌面更新后就可以點進去了
使用重啟桌面的方法,加快圖標的切換,1.會被系統禁用例如華為 Vivo Oppo則無法重啟桌面2.小米可以,但是還是會在1~3秒的時候關閉應用一次
僅僅修改的是啟動圖標,如果有快捷方式那么快捷方式不會發生改變,當然可以使用代碼動態更新快捷方式
無法動態加載網絡圖片,還是僅僅是本地資源修改的桌標
參考