框架源码地址: https://github.com/ximsfei/Android-skin-support
基本原理:
Android-skin-support 这个框架的主要思路是:
1. 更换LayoutInflaterFactory,对从xml布局文件要解析出来的 View,如果有相对应的支持换肤功能的控件(即实现SkinCompatSupportable接口的控件),就创建支持换肤功能的控件,然后把这个控件保存起来,如果没有就创建本来的控件(这个是不保存的);
2. 当收到换肤通知的时候,就调用所有保存里来的控件的applySkin() 进行换肤操作。
简单使用:
1. 初始化:
2. 更换SDcard 上的皮肤:
3. 换回默认皮肤:
接下来我们来看看源码。
一、简单的框架逻辑:
1. 在 Application 中注册 SkinActivityLifecycle 监听Activity生命周期;
2. 在 SkinActivityLifecycle 中 onActivityCreated() 设置
3. getSkinDelegate()中创建SkinCompatDelegate实例和activity绑定起来存入WeakHashMapmSkinDelegateMap中;
4. 在 SkinCompatDelegate 的回调 onCreateView()(和这个类的实例绑定的activity的所有view都会走这个函数) 判断view是否支持换肤功能,如果支持就存入 List> mSkinHelpers:
View view = createView(parent, name, context, attrs); //创建实现了SkinCompatSupportable的view实例
5. 在 SkinActivityLifecycle 的 onActivityResumed() 中创建皮肤更换观察者,并存入本实例的WeakHashMapmSkinObserverMap和 SkinCompatManager 的 ArrayListobservers:
创建皮肤观察者:
6. 在 SkinActivityLifecycle 的 onActivityDestroyed() 中移除与该activity绑定SkinCompatDelegate实例、SkinObserver:
二、更换皮肤逻辑:
1. 调用SkinCompatManager.getInstance().loadSkin("night", null, SkinCompatManager.SKIN_LOADER_STRATEGY_BUILD_IN); 启动一个异步任务SkinLoadTask:
2. 在异步任务 SkinLoadTask 的 doInBackground() 中根据strategy调用不同的皮肤包加载策略类的loadSkinInBackground(), 比如放在Assets皮肤包,这边就是从Assets复制皮肤包到getExternalCacheDir()目录下的skins目录下。
3. 在异步任务 SkinLoadTask 的 onPostExecute() 调用 SkinCompatManager 的 notifyUpdateSkin() 通知所有的 SkinObserver 皮肤更新了。
4. 回调 在 SkinActivityLifecycle 创建的 SkinObserver 实例的 updateSkin():
5. 调用 acitvity 对应的 SkinCompatDelegate 的 applySkin();
6. 在 SkinCompatDelegate 的 applySkin() 里面遍历mSkinHelpers(该activity所有的View实例)的applySkin();
7. 在 View 实例的 applySkin() 里面实现了关于这个控件换皮肤的操作。
三、LayoutInflaterFactory 创建解析出来的view相对应的支持换肤功能的控件
createViewFromHackInflater() 这里创建我们通过addHookInflater() 添加对应某个view替换成支持换肤的控件的规则,可以用来覆盖框架自带对应的支持换肤的控件,即对应某个view,创建我们的自己的控件,不创建框架自带的控件。
createViewFromInflater() 和上面不同,框架默认支持基本常用控件和一些V7包上的一些控件的替换,如果你需要添加其他控件的替换,可以使用addInflater()添加对应规则的View,但是覆盖不了框架默认支持的基本常用控件和一些V7包上的一些控件的替换控件。
基本常用控件的替换:
V7 包只替换了 Toolbar :