Android动态换肤(二、apk免安装插件方式)
在上一篇文章Android动态换肤(一、应用内置多套皮肤)中,我们了解到,动态换肤无非就是调用view的setBackgroundResource(R.drawable.id)等方法设置控件的背景或者文字等资源,跟踪源码看看这些方法都是怎么根据资源ID找到对应的资源的:
- View:
@RemotableViewMethod
public void setBackgroundResource(int resid) {
if (resid != 0 && resid == mBackgroundResource) {
return;
}
Drawable d = null;
if (resid != 0) {
d = mContext.getDrawable(resid);
}
setBackground(d);
mBackgroundResource = resid;
}
- mContext:
public final String getString(int resId) {
return getResources().getString(resId);
}
public final String getString(int resId, Object... formatArgs){
return getResources().getString(resId, formatArgs);
}
public final Drawable getDrawable(int id) {
return getResources().getDrawable(id, getTheme());
}
...
- Resources:
Resources中有一系列根据资源ID获取资源的方法,所以最关键的就是得到皮肤文件的Resources对象和资源ID。资源ID在皮肤apk打包的时候已经编译好,可以通过反射的方式获取R然后获取到资源ID。Android资源管理框架实际就是由AssetManager和Resources两个类来实现的,其中,Resources类可以根据ID来查找资源,而AssetManager类根据文件名来查找资源。事实上,如果一个资源ID对应的是一个文件,那么Resources类是先根据ID来找到资源文件名称,然后再将该文件名称交给AssetManager类来打开对应的文件的。
/**
* Create a new Resources object on top of an existing set of assets in an AssetManager.
*
* @param assets Previously created AssetManager.
* @param metrics Current display metrics to consider when
* selecting/computing resource values.
* @param config Desired device configuration to consider when
* selecting/computing resource values (optional).
*/
public Resources(AssetManager assets, DisplayMetrics metrics, Configuration config) {
this(assets, metrics, config, CompatibilityInfo.DEFAULT_COMPATIBILITY_INFO, null);
}
根据上面的构造方法,创建Resources对象,需要传入三个参数,分别是AssetManager对象(用于获取资源文件),DisplayMetrics对象(封装一些关于显示的通用信息,如显示大小,分辨率和字体),Configuration对象(专门用来描述手机设备上的配置信息。这些配置信息包括用户特定的配置项,也包括系统的动态设备配置);后面两个参数是关于Android设备的,同一台设备这两个对象封装内容也是相同的,所以直接传入当前应用中获取的即可,现在关键问题就是如何获取到皮肤apk的AssetManager对象。
/**
* Add an additional set of assets to the asset manager. This can be
* either a directory or ZIP file. Not for use by applications. Returns
* the cookie of the added asset, or 0 on failure.
* {@hide}
*/
public final int addAssetPath(String path) {
synchronized (this) {
int res = addAssetPathNative(path);
makeStringBlocks(mStringBlocks);
return res;
}
}
AssetManager中有一个addAssetPath方法,此方法将一个资源文件的路径添加到资源管理器中,这个资源文件的路径就是皮肤apk的路径;此方法被隐藏了,可以通过反射的方式调用。
到此为止,皮肤apk的资源管理器已经拿到了,代码如下:
public class BaseActivity extends Activity{
protected AssetManager mAssetManager; //资源管理器
protected Resources mResources; //资源
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
protected void loadResources(String dexPath) {
try {
AssetManager assetManager = AssetManager.class.newInstance();
Method addAssetPath = assetManager.getClass().getMethod("addAssetPath", String.class);
addAssetPath.invoke(assetManager, dexPath);
mAssetManager = assetManager;
} catch (Exception e) {
e.printStackTrace();
}
Resources superRes = super.getResources();
superRes.getDisplayMetrics();
superRes.getConfiguration();
mResources = new Resources(mAssetManager, superRes.getDisplayMetrics(),superRes.getConfiguration());
}
@Override
public Resources getResources() {
return mResources == null ? super.getResources() : mResources;
}
}
获取皮肤apk中的资源ID
Android中提供了DexClassLoader这个类加载器用来从.jar和.apk类型的文件内部加载classes.dex文件。通过这种方式可以用来执行非安装的程序代码,作为程序的一部分进行运行。获取资源ID有两种方式:
- 一种是在皮肤应用中编写工具类返回资源ID,然后通过反射调用此工具类:
public class UIUtil {
public static int getBtnBgId(){
return R.drawable.btn_selector;
}
public static int getMyActivityBg(){
return R.drawable.login_bg;
}
public static int getTilteString(){
return R.string.app_name;
}
....
}
@SuppressLint("NewApi")
private void loadOtherSkin(){
try {
Class clazz = classLoader.loadClass("com.openxu.skin.UIUtil");
Method method = clazz.getMethod("getTilteString");
titleID = (int) method.invoke(null);
Log.d(TAG, "获取标题ID:" + titleID);
method = clazz.getMethod("getBtnBgId");
btn_bg = (int) method.invoke(null);
Log.d(TAG, "获取按钮背景ID:" + btn_bg);
method = clazz.getMethod("getMyActivityBg");
activty_bg = (int) method.invoke(null);
Log.d(TAG, "获取背景ID:" + activty_bg);
showSkin();
} catch (Exception e) {
Log.e(TAG, "获取失败:" + Log.getStackTraceString(e));
}
}
- 另外一种方式是直接通过反射得到R文件中的各种资源类从而得到对应的资源ID:
@SuppressLint("NewApi")
private int getDrawableId(String name) {
try {
Class clazz = classLoader.loadClass("com.openxu.skin.R$drawable");
Field field = clazz.getField("ic_launcher");
int resId = (int) field.get(null);
return resId;
} catch (Exception e) {
Log.i("Loader", "error:" + Log.getStackTraceString(e));
}
return 0;
}
效果图:
下载源码后运行工程,在sd卡根目录新建skinchange文件夹,然后将工程下assets目录中的SkinChange2Skin.apk皮肤apk(zip为皮肤apk源码)拷贝到skinchange文件夹下,点击更换皮肤
源码下载:
Android动态换肤(二、apk免安装插件方式)的更多相关文章
- Android动态换肤(三、安装主题apk方式)
相比之前免安装的方式,这种方法需要用户下载并安装皮肤apk,程序写起来比免安装的要简单很多,像很多系统主题就是通过这种方式实现的. 这种方式的思路是,从所有已安装的应用程序中遍历出皮肤程序(根据特定包 ...
- Android动态换肤(一、应用内置多套皮肤)
动态换肤在很多android应用中都有使用,用户根据自己的喜好设置皮肤主题,可以增强用户使用应用的舒适度. Android换肤可以分为很多种,它们从使用方式,用户体验以及项目框架设计上体现了明显的差异 ...
- Android主题换肤实现
本系列文章主要是对一个Material Design的APP的深度解析,主要包括以下内容 基于Material Design Support Library作为项目整体框架.对应博文:Android ...
- Android主题换肤 无缝切换
2016年7月6日 更新:主题换肤库子项目地址:ThemeSkinning,让app集成换肤更加容易.欢迎star以及使用,提供改进意见. 更新日志: v1.3.0:增加一键切换切换字体(初版)v1. ...
- Android实现换肤功能(一)
上周有个朋友给建议说讲讲换肤吧,真巧这周公司的工作安排也有这个需求,换的地方之多之繁,让人伤神死了.正所谓磨刀不误砍柴工,先磨下刀,抽出一个工具类,写了个关于换肤的简单demo. Android中换肤 ...
- duilib入门之贴图描述、类html文本描述、动态换肤、Dll插件、资源打包
转载自duilib入门文档 贴图描述: Duilib的表现力丰富很大程度上得益于贴图描述的简单强大.Duilib的贴图描述分为简单模式和复杂模式两种. 简单模式使用文件名做为贴图描述内容,在这种方式下 ...
- element-ui 动态换肤
1.在安装好 element-ui@2.x 以后,首先安装sass-loader npm i sass-loader node-sass -D 2.安装 element-theme npm i ele ...
- hybird之web动态换肤实现
前言 最近在重构个hybird(原生的壳包着Web页面)的UI框架,进行到了做换肤功能的阶段,所以这里是我思考的解决的方法. 预想 目前实现换肤的功能无非就两种做法. 1.写几个皮肤文件,然后切换使用 ...
- CocoStudio基础教程(4)骨骼动画的动态换肤
1.概述 游戏中人物的状态会发生改变,而这种改变通常要通过局部的变化来表现出来.比如获得一件装备后人物形象的改变,或者战斗中武器.防具的损坏等.这些变化的实现就要通过动态换肤来实现. 2.运行到程序 ...
随机推荐
- [HNOI2010]STONE取石头游戏
题目描述 A 公司正在举办一个智力双人游戏比赛----取石子游戏,游戏的获胜者将会获得 A 公司提供的丰厚奖金,因此吸引了来自全国各地的许多聪明的选手前来参加比赛. 与经典的取石子游戏相比,A公司举办 ...
- [ZJOI2007]棋盘制作
题目描述 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源于易经的思想,棋盘是一个8*8大小的黑白相间的方阵,对应八八六十四卦,黑白对应阴阳. 而我们的 ...
- POJ - 3264:Balanced Lineup
ST表模版 #include<cstdio> #include<cstdlib> #include<algorithm> #include<cstring&g ...
- BZOJ4727 [POI2017]Turysta
这题太神了还是去看刺儿神题解吧. http://www.cnblogs.com/neighthorn/p/6538364.html #include <cstdio> #include & ...
- Unity3d 简单的小球沿贝塞尔曲线运动(适合场景漫游使用)
简单的小球沿贝塞尔曲线运动,适合场景漫游使用 贝塞尔曲线:(贝塞尔曲线的基本想法部分摘自http://blog.csdn.net/u010019717/article/details/4768 ...
- AspNetCoreApi 跨域处理
AspNetCoreApi 跨域处理 如果咱们有处理过MV5 跨域问题这个问题也不大. (1)为什么会出现跨域问题: 浏览器安全限制了前端脚本跨站点的访问资源,所以在调用WebApi 接口时不能成功 ...
- C语言程序设计第六次作业--循环结构(2)
(一)改错题 序列求和:输入一个正实数eps,计算序列部分和 1 - 1/4 + 1/7 - 1/10 + ... ,精确到最后一项的绝对值小于eps(保留6位小数). 输入输出样例: Input e ...
- 将 Hexo 个人博客同时部署到 GitHub 和 Coding 上
一.将个人博客托管到 GitHub 上 关于如何快速搭建自己的个人博客,如何完善自己的个人博客,什么是 GitHub ,如何将自己的博客代码托管到 GitHub 上面等等问题,我之前写过三篇文章已经做 ...
- a++与 ++a
a++先执行表达式再自增执行表达式使用a原值++a先自增再执行表达示执行表达式使用自增a例:int a=0printf("%d",a++); //输0,执行完a=1int a=0p ...
- 查询优化--ORDER BY查询优化
Mysql 系列文章主页 =============== ORDER BY 子句,尽量使用 Index 查询,避免使用 FileSort 排序 尽可能在索引列上完成排序操作,遵照索引的最佳左前缀原则 ...