android动态换肤使用本地资源原理分析

2024-05-04 12:32

本文主要是介绍android动态换肤使用本地资源原理分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

大致原理:

在application里面注册所有activity回调这样可以实现很少的改动侵入性

  1. 给LayoutFactory设置自己的factory2,工厂2 使activity在setContentView调用inflate的时候触发自己的factory的创建view方法.,为什么可以呢?因为LayoutFactory.from(this)当前activity填充的时候调用多次实际上还是同样的对象,所以可以这么简单的进行了hook 不需要进行更高级的hook from返回值类似的方法操作。

  2. 创建的时候根据属性和节点判断是否需要更换皮肤,比如#开头引用的可以忽略,对于属性 ```?``修饰或者是@引用的进行记录,在view构造方法创建完毕之后设置一次皮肤就完成了更改. 先通过原来apk的id转换为资源名称然后 用这个资源名称在自己构建的Resource里面进行加载如果能加载就用自己的,否则还是用系统的.其实如果没加载到干脆不操作也是可以的.

  3. 比如对背景颜色进行换肤,实际上一个view的setBackground方法也许会调用2次,因为第一次是构造的时候new ImageView(context,attrs)的时候进行了一次初始化,构造完成之后更换加载默认皮肤又进行了一次替换.

大致经典代码:

创建自己的Resource

AssetManager assetManager = AssetManager.class.newInstance();// 添加资源进入资源管理器Method addAssetPath = assetManager.getClass().getMethod("addAssetPath", String.class);addAssetPath.setAccessible(true);addAssetPath.invoke(assetManager, path);Resources resources = application.getResources();Resources skinResource = new Resources(assetManager, resources.getDisplayMetrics(),resources.getConfiguration());PackageManager mPm = application.getPackageManager();PackageInfo info = mPm.getPackageArchiveInfo(path, PackageManager.GET_ACTIVITIES);String packageName = info.packageName;} catch (Exception e) {e.printStackTrace();}

上面的方法是拿到皮肤apk 然后自己构造一个Resources.并调用资源管理器的addAssetPath添加进去,避免不合法,但是此时id是不相同的,

view的属性节点操作

/**** @param view* @param attrs*/public void  parseViewInfo(View view, AttributeSet attrs) {List<SkinPair> skinPairs = new ArrayList<>();for (int i = 0; i < attrs.getAttributeCount(); i++) {String attributeName = attrs.getAttributeName(i);if (mAttributes.contains(attributeName)) {String attributeValue = attrs.getAttributeValue(i);    if (attributeValue.startsWith("#")) {continue;}//资源idint resId;if (attributeValue.startsWith("?")) {//得到属性id 解析成具体值idint attrId = Integer.parseInt(attributeValue.substring(1));resId = SkinThemeUtils.getResIdByAttr(view.getContext(), new int[]{attrId})[0];} else {// @694886526resId = Integer.parseInt(attributeValue.substring(1));}if (resId != 0) {//对于匹配的成功找到的进行添加 方便更换}}}}

换肤的操作

换肤的时候只需要查找是否匹配,成功通过资源名找到资源id才进行替换

public int getIdentifier(int resId) {if (isDefaultSkin) {return resId;}//R.drawable.ic_launcherString resName = mAppResources.getResourceEntryName(resId);//ic_launcherString resType = mAppResources.getResourceTypeName(resId);//如 drawable color int skinId = mSkinResources.getIdentifier(resName, resType, mSkinPkgName);return skinId;}
public int getColor(int resId) {if (isDefaultSkin) {return mAppResources.getColor(resId);}int skinId = getIdentifier(resId);if (skinId == 0) {return mAppResources.getColor(resId);}return mSkinResources.getColor(skinId);}

上面的方法拿颜色值是先判断是否启用皮肤如果没有启用皮肤那么根据getIdentifier(resid)方法先传递原始属性id得获取资源名称,然后把资源名称和类型传递进去再取调用自己创建的皮肤Resources

mSkinResources.getIdentifier(resName, resType, mSkinPkgName);

如果能拿到自然就从自己的皮肤资源里面获取颜色值了,那么对于属性替换又是如何操作呢?

public static int[] getResIdFromAttrs(Context context, int[] attrs) {int[] resIds = new int[attrs.length];TypedArray typedArray = context.obtainStyledAttributes(attrs);for (int i = 0; i < typedArray.length(); i++) {resIds[i] = typedArray.getResourceId(i, 0);}typedArray.recycle();return resIds;}

这篇关于android动态换肤使用本地资源原理分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/959201

相关文章

使用Python删除Excel中的行列和单元格示例详解

《使用Python删除Excel中的行列和单元格示例详解》在处理Excel数据时,删除不需要的行、列或单元格是一项常见且必要的操作,本文将使用Python脚本实现对Excel表格的高效自动化处理,感兴... 目录开发环境准备使用 python 删除 Excphpel 表格中的行删除特定行删除空白行删除含指定

深入理解Go语言中二维切片的使用

《深入理解Go语言中二维切片的使用》本文深入讲解了Go语言中二维切片的概念与应用,用于表示矩阵、表格等二维数据结构,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录引言二维切片的基本概念定义创建二维切片二维切片的操作访问元素修改元素遍历二维切片二维切片的动态调整追加行动态

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串

prometheus如何使用pushgateway监控网路丢包

《prometheus如何使用pushgateway监控网路丢包》:本文主要介绍prometheus如何使用pushgateway监控网路丢包问题,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录监控网路丢包脚本数据图表总结监控网路丢包脚本[root@gtcq-gt-monitor-prome

Python通用唯一标识符模块uuid使用案例详解

《Python通用唯一标识符模块uuid使用案例详解》Pythonuuid模块用于生成128位全局唯一标识符,支持UUID1-5版本,适用于分布式系统、数据库主键等场景,需注意隐私、碰撞概率及存储优... 目录简介核心功能1. UUID版本2. UUID属性3. 命名空间使用场景1. 生成唯一标识符2. 数

SpringBoot中如何使用Assert进行断言校验

《SpringBoot中如何使用Assert进行断言校验》Java提供了内置的assert机制,而Spring框架也提供了更强大的Assert工具类来帮助开发者进行参数校验和状态检查,下... 目录前言一、Java 原生assert简介1.1 使用方式1.2 示例代码1.3 优缺点分析二、Spring Fr

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class

SpringBoot线程池配置使用示例详解

《SpringBoot线程池配置使用示例详解》SpringBoot集成@Async注解,支持线程池参数配置(核心数、队列容量、拒绝策略等)及生命周期管理,结合监控与任务装饰器,提升异步处理效率与系统... 目录一、核心特性二、添加依赖三、参数详解四、配置线程池五、应用实践代码说明拒绝策略(Rejected