换肤

Android换肤功能

  • 什么是换肤?
    • app的皮肤,比如说黑夜模式,切换之后整体风格改变成以黑色为主题色
  • 换了什么?
    • 背景、颜色、图片、字体等等
  • 换肤的原理
    • 加载另一个apk中的相同名字的图片颜色等资源
  • 思路
    • 监察当前apk中xml生成的加载过程
    • 拿到所有具有换肤潜质的控件(包括自定义控件)
    • 这些控件都通过统一的管理者(SkinManager)来设置颜色或者背景资源等
    • 下载换肤apk、获取资源
    • ** 开始换肤**
  • 代码实现
    • 新建BaseActivity通过LayoutInflaterCompat.setFactory2(?,?)监听xml的生成过程,里面需要我们传两个参数,第一个很简单直接 getLayoutInflater() 获取就可以,第二个参数需要我们传递一个 LayoutInflater.Factory2 这个类需要我们自己复写一遍好收集换肤的控件
        skinFactory = new SkinFactory();
        LayoutInflaterCompat.setFactory2(getLayoutInflater(), skinFactory);
  • 在SkinFactory中实现 LayoutInflater.Factory2 接口,在onCreateView方法中进行控件的收集

这里的name就是我们获取到的控件名,需要注意的是,自定义控件是全名比如 com.leary.MyTextVIew 二系统控件不是全名,所以需要们把它补充完整

  private static final String[] prefixList = {"android.widget.", "android.view.", "android.webkit."};

  @Override
  public View onCreateView(View parent, String name, Context context, AttributeSet attrs) {
    //收集需要换肤的控件
    View view = null;

    if (name.contains(".")) {// 自定义控件
        view = createView(context, attrs, name);
    } else {//系统控件
        for (String prefix : prefixList) {
            view = createView(context, attrs, prefix + name);
            if (view != null) {
                break;
            }
        }
    }
    if (view != null) {
        parseSkinView(context, attrs,view);
    }
    return view;
  }

createView需要用到反射,我贴一下代码

      Class viewClazz = context.getClassLoader().loadClass(name);
            Constructor<? extends View> constructor = viewClazz.getConstructor(
                    new Class[]{Context.class, AttributeSet.class});
            return constructor.newInstance(context, attrs);

当View被创建完成,这个时候就需要们完成收集了

      private void parseSkinView(Context context, AttributeSet attrs, View view) {
        List<SkinItem> list = new ArrayList<>();
        for (int i=0;i<attrs.getAttributeCount();i++) {
            String attrName = attrs.getAttributeName(i);
            String attrValue = attrs.getAttributeValue(i);
            if ("textColor".equals(attrName) || "background".equals(attrName)) {
                //可换肤的控件
                int id = Integer.parseInt(attrValue.substring(1));
                String entry_name = context.getResources().getResourceEntryName(id);
                String typeName = context.getResources().getResourceTypeName(id);
                SkinItem skinItem = new SkinItem(attrName, entry_name, typeName, id);
                list.add(skinItem);
            }
        }
        if (!list.isEmpty()) {
            SkinView skinView = new SkinView(view, list);
            cacheList.add(skinView);
            //xml加载过程中换肤
            skinView.apply();
        }
      }

这里面SkinItem是收集的控件的属性,也不多解释看图吧


image

SkinView就是封装当前View和所有控件集合的对象

换肤的时候从SkinManger里边去拿

      public void apply() {
        //应用所有的换肤
        for (SkinItem skinItem : list) {
            if ("background".equals(skinItem.getAttrName())) {
                if ("color".equals(skinItem.getAttrType())) {
               view.setBackgroundColor(SkinManager.getInstance().getColor(skinItem.getAttrId()));
                } else if ("drawable".equals(skinItem.getAttrType())) {
                    view.setBackgroundDrawable(SkinManager.getInstance().getDrawable(skinItem.getAttrId()));
                }
            }
        }
      }

SkinManager只需要干一个事情

  • 获取资源(当前apk或者其他apk的资源),里面还会用到反射
    public class SkinManager {
    private static final SkinManager ourInstance = new SkinManager();

    public static SkinManager getInstance() {
        return ourInstance;
    }

    private SkinManager() {
    }

    //apk中的resources
    private Resources skinResources;
    private Context context;
    //皮肤apk的包名
    private String skinPackage;

    public void init(Context context) {
        this.context = context.getApplicationContext();
    }

    public Resources getSkinResources() {
        return skinResources;
    }

    //获取resId
    public int getColor(int resId) {
        if (skinResources == null) {
            return ContextCompat.getColor(context, resId);
        }
        String resName = context.getResources().getResourceEntryName(resId);
        int skinId = skinResources.getIdentifier(resName, "color", skinPackage);
        if (skinId == 0) {
            return ContextCompat.getColor(context, resId);
        }
        if (resId == skinId) {
            Log.e("leary", resName + " id一样 " + skinPackage);
        }
        return skinResources.getColor(skinId);
    }

    public Drawable getDrawable(int resId) {
        if (skinResources != null) {
            String resName = context.getResources().getResourceEntryName(resId);
            int skinId = skinResources.getIdentifier(resName, "drawable", skinPackage);
            if (skinId == 0) {
                return ContextCompat.getDrawable(context, resId);
            }
            return skinResources.getDrawable(skinId);
        }
        return ContextCompat.getDrawable(context, resId);
    }

    //加载apk
    public void loadApk(String path) {
        try {
            AssetManager assetManager = AssetManager.class.newInstance();
            Method addAssetPath = AssetManager.class.getMethod("addAssetPath", String.class);
            addAssetPath.invoke(assetManager, path);
            skinResources = new Resources(assetManager, context.getResources().getDisplayMetrics(),
                    context.getResources().getConfiguration());
            PackageManager packageManager = context.getPackageManager();
            //拿到皮肤的包名
            skinPackage = packageManager.getPackageArchiveInfo(path, PackageManager.GET_ACTIVITIES).packageName;
        } catch (InstantiationException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }


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

推荐阅读更多精彩内容