android换肤原理解析

: )


**En**

首先来说说应用场景

  • app里面的一些控件的属性(字体大小,字体颜色,背景..)需要根据皮肤包里面的资源随意切换
  • 这里有一个要求就是app里面的资源名字要和皮肤包里面的资源名字一样(控件里面必须使用资源id来访问,你都不用资源id来访问你还好意思换肤 -_-!

需要解决的问题

  • 皮肤包怎么生成
  • 如何通过资源名字去访问皮肤包里面的资源,并将资源拿出来使用
  • 如何比较方便的使我们将获取到的资源设置到控件上面去

皮肤包的生成

  • 其实很简单,就是我们重新建立一个项目(这个项目里面的资源名字和需要换肤的项目的资源名字是对应的就可以),记住我们是通过名字去获取资源,不是id,不是id(名字对应就可以了)

  • 这个是测试的一个布局

测试用布局
  • 这个界面包含一个 标题,一个图片,一个名字和一个换肤的按钮

    • 标题的颜色 <color name="title_text_color">#0ff</color>
    • 图片的资源 android:src="@mipmap/demo"
    • 底部的文字 android:text="@string/create_name" (Dapi)
  • 重新建立一个工程,也添加和上面名字相同的资源

    • "<color name="title_text_color">#000</color>" (黑色)
    • "<string name="create_name">大批</string>" (大批)
  • 将皮肤项目编译成apk(这个apk的后缀名就随意了),并传到手机上面去(传到一个你等会能访问到的地方就行)

我用的是adb push

如何通过资源名字来获取到上面生成的皮肤包资源

  • Android访问资源使用的是Resources这个类,但是程序里面通过getContext获取到的Resources实例实际上是对应程序本来的资源的实例,也就是说这个实例只能加载app里面的资源,想要加载皮肤包里面的就不行了
  • 自己构造一个Resources(这个Resources指向的资源就是我们的皮肤包)
  • 看看Resources的构造方法,可以看到主要是需要一个AssetManager
public Resources(AssetManager assets, DisplayMetrics metrics, Configuration config) {
        this(null);
        mResourcesImpl = new ResourcesImpl(assets, metrics, config, new DisplayAdjustments());
    }
  • 构造一个指向皮肤包的AssetManager,但是这个AssetManager是不能直接new出来的,这里就使用反射来实例化了
AssetManager assetManager = AssetManager.class.newInstance();
  • AssetManager有一个addAssetPath方法可以指定资源的位置,可惜这个也只能用反射来调用
Method addAssetPath = assetManager.getClass().getMethod("addAssetPath", String.class);
        addAssetPath.invoke(assetManager, filePath);
  • 再来看看Resources的其他两个参数,一个是DisplayMetrics,一个是Configuration,这两的就可以直接使用app原来的Resources里面的就可以

  • 构造皮肤包的**Resources **代码

/**
         * 皮肤包的位置
         */
        String filePath = Environment.getExternalStorageDirectory() +
                File.separator + "demo.skin";

        AssetManager assetManager = AssetManager.class.newInstance();
        Method addAssetPath = assetManager.getClass().getMethod("addAssetPath", String.class);
        addAssetPath.invoke(assetManager, filePath);

        Resources resources = new Resources(
                assetManager,
                orginResources.getDisplayMetrics(),
                orginResources.getConfiguration()
        );
  • 根据皮肤包的Resources 去访问里面的资源(这里就比较简单了,通过Resources 访问资源一般都是需要一个资源id,所以我们先将资源名字转化成对应的id,然后通过id去访问资源就可以了)

  • 这里就是仅仅先访问前面定义好的资源,还有一个需要注意的是这里还需要皮肤包的包名

private static final String SKIN_PAGNAME = "com.suse.skindemo";

    private String getNameString(Resources resources){
        int id = resources.getIdentifier("create_name","string",SKIN_PAGNAME);
        return resources.getString(id);
    }

    private ColorStateList getTitleColor(Resources resources){
        int id = resources.getIdentifier("title_text_color","color",SKIN_PAGNAME);
        return resources.getColorStateList(id);
    }

    private Drawable getContentDrawable(Resources resources){
        int id = resources.getIdentifier("demo","mipmap",SKIN_PAGNAME);
        return resources.getDrawable(id);
    }
  • 实现的效果
访问皮肤包的资源

如何比较方便的使用?

  • 上面的实现过程是我们通过我们自己构造的Resources对象访问到资源,并调用控件的方法将获取到的资源设置过去(这样的话每个需要换肤的控件都有一个设置资源的逻辑 what??)

  • LayoutInflater.Factory来简化操作(LayoutInflator在创建控件的时候就是通过这个Factory来创建的),这里简单介绍一下思路,就是自己定义一个Factory,在LayoutInflater创建view的时候就会调用我们自己的Factory,我们可以根据控件的属性信息来判断是否需要换肤

  • 这里还有一个需要注意的是在Factory回调的时候需要约定什么样的属性需要换肤(可以根据名字的前缀之类的)

  • 贴一个部分代码吧(具体代码最后会给出两个开源项目)

LayoutInflater inflater = getLayoutInflater();
        LayoutInflaterCompat.setFactory(inflater, new LayoutInflaterFactory() {
            @Override
            public View onCreateView(View parent, String name, Context context, AttributeSet attrs) {

                LayoutInflater layoutInflater = getLayoutInflater();
                AppCompatDelegate delegate = getDelegate();
                View view = null;
                try
                {
                    //public View createView
                    // (View parent, final String name, @NonNull Context context, @NonNull AttributeSet attrs)
                    if (sCreateViewMethod == null)
                    {
                        Method methodOnCreateView = delegate.getClass().getMethod("createView", sCreateViewSignature);
                        sCreateViewMethod = methodOnCreateView;
                    }
                    Object object = sCreateViewMethod.invoke(delegate, parent, name, context, attrs);
                    view = (View) object;
                } catch (NoSuchMethodException e)
                {
                    e.printStackTrace();
                } catch (InvocationTargetException e)
                {
                    e.printStackTrace();
                } catch (IllegalAccessException e)
                {
                    e.printStackTrace();
                }

                if (view == null)
                {
                    view = createViewFromTag(context, name, attrs);
                }
                return view;
            }
        });

最后给出两个开源项目吧,思路都是来自这两个项目

https://github.com/hongyangAndroid/ChangeSkin
https://github.com/fengjundev/Android-Skin-Loader


Nothing is certain in this life. The only thing i know for sure is that. I love you and my life. That is the only thing i know. have a good day

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

推荐阅读更多精彩内容

  • 前言 Android换肤技术已经是很久之前就已经被成熟使用的技术了,然而我最近才在学习和接触热修复的时候才看到。在...
    静默加载阅读 3,062评论 1 8
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,430评论 25 707
  • 前言: 本文主要讲述如何在项目中,在不重启应用的情况下,实现动态换肤的效果。换肤这块做的比较好的,有网易云音乐,q...
    Yagami3zZ阅读 13,588评论 5 51
  • 当我 还是个小孩子 那些单纯的日子 却离我越来越远 孤独 忍藏在卑微里的花蕊里 却又为何只争朝夕的开放着 那些迎着...
    河北人张鹏程阅读 170评论 0 0
  • 其实,我打下上面的题目时,我是还不知道要写些什么的。只知道自己要写,还要完全一个打赌,不能输那一块钱。 如果我有八...
    力牧阅读 182评论 1 1