Android 常用換膚方式以及原理分析
阿新 • • 發佈:2018-11-22
Android 換膚
常用方法
1.通過Theme切換主題
通過在setContentView之前設定Theme實現主題切換。
在styles.xml定義一個夜間主題和白天主題:
<style name="LightTheme" parent="Theme.AppCompat.Light.DarkActionBar"> <item name="colorPrimary">@color/colorPrimary</item> <item name="colorPrimaryDark">@color/colorPrimaryDark</item> <item name="colorAccent">@color/colorAccent</item> <!--主題背景--> <item name="backgroundTheme">@color/white</item> </style> <style name="BlackTheme" parent="Theme.AppCompat.Light.DarkActionBar"> <item name="colorPrimary">@color/colorPrimary</item> <item name="colorPrimaryDark">@color/colorPrimaryDark</item> <item name="colorAccent">@color/colorAccent</item> <!--主題背景--> <item name="backgroundTheme">@color/dark</item> </style>
設定主要切換主題View的背景:
<android.support.constraint.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:layout_width="match_parent" android:layout_height="match_parent" android:background="?attr/backgroundTheme" tools:context=".MainActivity"> <Button android:id="@+id/btn" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="切換主題" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintLeft_toLeftOf="parent" app:layout_constraintRight_toRightOf="parent" app:layout_constraintTop_toTopOf="parent" /> </android.support.constraint.ConstraintLayout>
切換主題:
通過呼叫setTheme()
@Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setTheme(R.style.BlackTheme); setContentView(R.layout.activity_main); } finish(); Intent intent = getIntent(); intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_CLEAR_TASK); startActivity(intent); overridePendingTransition(0, 0);
效果如下:
2.通過AssetManager切換主題
下載面板包,通過AssetManager載入面板包裡面的資原始檔,實現資源替換。
ClassLoader
Android可以通過classloader獲取已安裝apk或者未安裝apk、dex、jar的context物件,從而通過反射去獲取Class、資原始檔等。
載入已安裝應用的資源
//獲取已安裝app的context物件
Context context = ctx.getApplicationContext().createPackageContext("com.noob.resourcesapp", Context.CONTEXT_INCLUDE_CODE | Context.CONTEXT_IGNORE_SECURITY);
//獲取已安裝app的resources物件
Resources resources = context.getResources();
//通過resources獲取classloader,反射獲取R.class
Class aClass = context.getClassLoader().loadClass("com.noob.resourcesapp.R$drawable");
int resId = (int) aClass.getField("icon_collect").get(null);
imageView.setImageDrawable(resources.getDrawable(id));
載入未安裝應用的資源
String apkPath = Environment.getExternalStorageDirectory().getAbsolutePath() + "/test.apk";
//通過反射獲取未安裝apk的AssetManager
AssetManager assetManager = AssetManager.class.newInstance();
//通過反射增加資源路徑
Method method = assetManager.getClass().getMethod("addAssetPath", String.class);
method.invoke(assetManager, apkPath);
File dexDir = ctx.getDir("dex", Context.MODE_PRIVATE);
if (!dexDir.exists()) {
dexDir.mkdir();
}
//獲取未安裝apk的Resources
Resources resources = new Resources(assetManager, ctx.getResources().getDisplayMetrics(),
ctx.getResources().getConfiguration());
//獲取未安裝apk的ClassLoader
ClassLoader classLoader = new DexClassLoader(apkPath, dexDir.getAbsolutePath(), null, ctx.getClassLoader());
//反射獲取class
Class aClass = classLoader.loadClass("com.noob.resourcesapp.R$drawable");
int id = (int) aClass.getField("icon_collect").get(null);
imageView.setImageDrawable(resources.getDrawable(id));
LayoutInflater.Factory
分析setContentView原始碼
LayoutInflater.Factory是如何被呼叫的
setContentView最終呼叫了inflate方法,我們來看一下inflate方法的原始碼
inflate最終呼叫了createViewFromTag方法來建立View,在這之中用到了factory,如果factory存在就用factory建立物件,如果不存在就由系統自己去建立。
我們在setContentView之前呼叫測試程式碼
測試程式碼:
LayoutInflater.from(this).setFactory(new LayoutInflater.Factory() {
@Override
public View onCreateView(String name, Context context, AttributeSet attrs) {
Log.e("MainActivity", "name :" + name);
int count = attrs.getAttributeCount();
for (int i = 0; i < count; i++) {
Log.e("MainActivity", "AttributeName :" + attrs.getAttributeName(i) + "AttributeValue :"+ attrs.getAttributeValue(i));
}
return null;
}
});
log日誌:
結果發現我們可以獲取一個layout的所有View,此時我們就可以對View進行面板切換效果。
通過AssetManager切換主題總結
通過AssetManager和LayoutInflater.Factory配合就可以達到呼叫外部資源獲取面板的方法。如果想要動態更新,只需要把需要動態更新的View存起來,去遍歷設定面板,或者用eventBus去通知也可以。
對比
上述兩種方法是市面上大多數換膚框架的實現原理。
通過Theme切換主題:
優點:實現簡單,配置簡單
缺點:需要重啟應用;是固定面板,不能動態切換
通過AssetManager切換主題:
優點:不需要重啟應用;可以動態載入主題,用於盈利
缺點:實現較為複雜;面板包比較佔資源