插件式换肤框架搭建 - 插件式换肤框架的完善

1. 概述


基于插件式换肤框架搭建 - 资源加载源码分析插件式换肤框架搭建 - setContentView源码阅读前两篇文章,那么目前我们不仅可以从另外一个插件皮肤包中获取资源了而且还可以去拦截系统View的创建,那么现在我们只要写点代码就可以达到无缝换肤的效果了。

GIF.gif

所有分享大纲:2017Android进阶之路与你同行

视频讲解地址:http://pan.baidu.com/s/1nvv2Nln

2. Hook拦截View的创建


前面讲解模板设计模式构建BaseActivity的时候,因为想到了框架的可扩展性所以就特意留了一层BaseSkinActivity,我们就在这里面去拦截和创建View就好,前提是要兼容tint等等的一些功能,下面的这些代码我们都是看源码得到的,google工程师写的基本不会有问题,看他们的源码考虑得很周到,还是可以借鉴一下的:

public class BaseSkinActivity extends AppCompatActivity implements LayoutInflaterFactory {
    // 创建View的Inflater
    private SkinCompatViewInflater mSkinCompatViewInflater;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // 给系统的LayoutInflater 设置Factory可以仿照系统的源码去写
        LayoutInflater layoutInflater = LayoutInflater.from(this);
        LayoutInflaterCompat.setFactory(layoutInflater, this);
        super.onCreate(savedInstanceState);
    }

    @Override
    public View onCreateView(View parent, final String name, @NonNull Context context,
                             @NonNull AttributeSet attrs) {
        final boolean isPre21 = Build.VERSION.SDK_INT < 21;

        if (mSkinCompatViewInflater == null) {
            mSkinCompatViewInflater = new SkinCompatViewInflater();
        }

        // We only want the View to inherit its context if we're running pre-v21
        final boolean inheritContext = isPre21 && shouldInheritContext((ViewParent) parent);

        View view = mSkinCompatViewInflater.createView(parent, name, context, attrs, inheritContext,
                isPre21, /* Only read android:theme pre-L (L+ handles this anyway) */
                true, /* Read read app:theme as a fallback at all times for legacy reasons */
                VectorEnabledTintResources.shouldBeUsed() /* Only tint wrap the context if enabled */
        );

        if (view != null) {
            List<SkinAttr> skinAttrs = SkinSupport.getSkinView(context, attrs);
            if (skinAttrs.size() != 0) {
                addSkinManager(view, skinAttrs);
            }
        }
        return view;
    }

    // 添加到皮肤管理
    private void addSkinManager(View view, List<SkinAttr> skinAttrs) {
        SkinView skinView = new SkinView(view, skinAttrs);

        List<SkinView> skinViews = SkinManager.getInstance().getSkinViews(this);

        if (skinViews == null) {
            skinViews = new ArrayList<>();
            SkinManager.getInstance().registerSkinView(skinViews, this);
        }
        skinViews.add(skinView);
        // 如果需要换肤
        if(SkinManager.getInstance().needChangeSkin()){
            SkinManager.getInstance().changeSkin(skinView);
        }
    }

    // 从系统源码贴的,google工程师写的基本不会有问题
    private boolean shouldInheritContext(ViewParent parent) {
        if (parent == null) {
            // The initial parent is null so just return false
            return false;
        }
        final View windowDecor = getWindow().getDecorView();
        while (true) {
            if (parent == null) {
                // Bingo. We've hit a view which has a null parent before being terminated from
                // the loop. This is (most probably) because it's the root view in an inflation
                // call, therefore we should inherit. This works as the inflated layout is only
                // added to the hierarchy at the end of the inflate() call.
                return true;
            } else if (parent == windowDecor || !(parent instanceof View)
                    || ViewCompat.isAttachedToWindow((View) parent)) {
                // We have either hit the window's decor view, a parent which isn't a View
                // (i.e. ViewRootImpl), or an attached view, so we know that the original parent
                // is currently added to the view hierarchy. This means that it has not be
                // inflated in the current inflate() call and we should not inherit the context.
                return false;
            }
            parent = parent.getParent();
        }
    }
}

3. 完善SkinManager这个管理类


在网上也看了很多第三方的换肤框架都还蛮好的,但是总感觉达不到自己想要的效果。其实之所以自己写是因为当时我真的想把电脑一脚踢了,就那么一个Bug自己搞了好几天没搞定,从来没有那么痛苦过。自己写也就没那么大的局限性,也不需要关注配置什么的,直接一行解决问题。

 /**
 * Created by Darren on 2017/3/20.
 * Email: 240336124@qq.com
 * Description: 皮肤的管理类
 */
public class SkinManager {
    private static SkinManager mInstance;
    private SkinResources mSkinResources;
    private Context mContext;

    private Map<ISkinChangeListener, List<SkinView>> mSkinViews = new HashMap<>();
    // 把View交给它管理

    private SkinManager() {
    }

    /**
     * 初始化 一般都在Application中配置
     * @param context
     */
    public void init(Context context) {
        this.mContext = context.getApplicationContext();
        String skinPath = SkinPreUtils.getInstance(mContext).getSkinPath();

        if (TextUtils.isEmpty(skinPath)) {
            return;
        }

        File skinFile = new File(skinPath);
        if (!skinFile.exists()) {
            clearSkinInfo();
            return;
        }

        initSkinResource(skinPath);
    }

    static {
        mInstance = new SkinManager();
    }

    public static SkinManager getInstance() {
        return mInstance;
    }


    /**
     * 加载皮肤
     *
     * @param path 当前皮肤的路径
     */
    public int loadSkin(String path) {
        String currentSkinPath = SkinPreUtils.getInstance(mContext).getSkinPath();
        if (currentSkinPath.equals(path)) {
            return SkinConfig.SKIN_LOADED;
        }

        File skinFile = new File(path);
        if(!skinFile.exists()){
            return SkinConfig.SKIN_PATH_ERROR;
        }

        // 判断签名是否正确,后面增量更新再说
        initSkinResource(path);
        changeSkin(path);
        saveSkinInfo(path);
        // 加载成功
        return SkinConfig.SKIN_LOAD_SUCCESS;
    }

    /**
     * 切换皮肤
     *
     * @param path 当前皮肤的路径
     */
    private void changeSkin(String path) {
        for (ISkinChangeListener skinChangeListener : mSkinViews.keySet()) {
            List<SkinView> skinViews = mSkinViews.get(skinChangeListener);
            for (SkinView skinView : skinViews) {
                skinView.skin();
            }
            skinChangeListener.changeSkin(path);
        }
    }

    /**
     * 初始化皮肤的Resource
     *
     * @param path
     */
    private void initSkinResource(String path) {
        mSkinResources = new SkinResources(mContext, path);
    }

    /**
     * 保存当前皮肤的信息
     *
     * @param path 当前皮肤的路径
     */
    private void saveSkinInfo(String path) {
        SkinPreUtils.getInstance(mContext).saveSkinPath(path);
    }

    /**
     * 恢复默认皮肤
     */
    public void restoreDefault() {
        String currentSkinPath = SkinPreUtils.getInstance(mContext).getSkinPath();
        if (TextUtils.isEmpty(currentSkinPath)) {
            return;
        }

        String path = mContext.getPackageResourcePath();
        initSkinResource(path);
        changeSkin(path);
        clearSkinInfo();
    }

    /**
     * 清空皮肤信息
     */
    private void clearSkinInfo() {
        SkinPreUtils.getInstance(mContext).clearSkinInfo();
    }

    public SkinResources getSkinResources() {
        return mSkinResources;
    }

    /**
     * 注册监听回调
     */
    public void register(List<SkinView> skinViews, ISkinChangeListener skinChangeListener) {
        mSkinViews.put(skinChangeListener, skinViews);
    }

    public List<SkinView> getSkinViews(Activity activity) {
        return mSkinViews.get(activity);
    }

    public boolean isChangeSkin() {
        return SkinPreUtils.getInstance(mContext).needChangeSkin();
    }

    public void changeSkin(SkinView skinView) {
        skinView.skin();
    }

    /**
     * 移除回调,怕引起内存泄露
     */
    public void unregister(ISkinChangeListener skinChangeListener) {
        mSkinViews.remove(skinChangeListener);
    }
}

4. 最后的换肤使用和测试


前提是我们所有的Activity必须继承自BaseSkinActivity,然后在任何一个地方都可以用SkinManager的loadSkin()方法就可以达到无缝切换不需要重启app,目前虽然整个换肤框架只有十几个类但是有事没事就可以优化优化或者增加点功能什么的。奇葩的自定义View那就只能自己在Activity做一丁点手脚,这是目前还没有办法解决的问题。

public class MainActivity extends BaseSkinActivity{

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    public void skin(View view) {
        // 1. 首先去网上下载资源皮肤
        // 2. 下载后保存在本地
        String skinPath = Environment.getExternalStorageDirectory().getAbsolutePath()
                + File.separator + "skin1.skin";

        // 3. 调用该方法去加载皮肤 不需要重启
        int result = SkinManager.getInstance().loadSkin(skinPath);
        // 可以判断result是否换肤成功,不同的状态对应不同的原因
    }

    public void skin1(View view) {
        // 恢复默认皮肤
        SkinManager.getInstance().restoreDefault();
    }

    // 跳转activity
    public void skin2(View view) {
        Intent intent = new Intent(this, MainActivity.class);
        startActivity(intent);
    }
}

所有分享大纲:2017Android进阶之路与你同行

视频讲解地址:http://pan.baidu.com/s/1nvv2Nln

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,240评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,328评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,182评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,121评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,135评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,093评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,013评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,854评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,295评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,513评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,678评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,398评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,989评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,636评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,801评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,657评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,558评论 2 352

推荐阅读更多精彩内容