Android 使用Overlay现实主题切换

最近项目上,想做一个主题切换的功能,整理了一下发布出来,主要使用的是IOverlayManager,大体思路如下:

1、想切换的应用,各自做overlay apk(简称皮肤包)

2、将overlay apk push 到vendor/overlay目录下(如果没有这个目录,push到system/app里面也可以),这个主要目的是为了让overlay 找到这个overlay apk。

3、重启设备

4、点击切换主题

5、各个应用重走生命周期,然后显示出来新的资源

这个主题切换的原理,我自己感觉,就是将overlay apk里面的资源替换到原来apk里面,然后重新加载。

一、app准备,制作overlay apk皮肤包

1、AS内新建项目 文件命名方式:xxxx.Red/xxxx.Young/xxxx.Gold

例如:LocalSetting 新建替换资源的模块名可命名为 LocalSetting.Red /LocalSetting.Young/LocalSetting.Gold,其实就和新建的app 项目一样。

仅保留AndroidManifest.xml 和res目录下仅保留需要替换的资源信息(values下的colors.xml , strings.xml , dimens.xml themes.xml等 和 drawables/xml目录下的资源, 不支持替换布局文件),想替换那个文件,就在对应的目录下用相同的资源名字放进去。

下图是launcher的主题app目录。

AndroidManifest.xml仅包含以下信息即可,其中targetPackage为要替换资源的app包名。

java 复制代码
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.test.launcher.young">

    <overlay
        android:priority="9"
        android:targetPackage="com.test.launcher" />
</manifest>

2、生成overlay apk;

overlay apk 中 AndroidManifest.xml 包名命名方式:xxxx.red/xxxx.young/xxxx.gold,如上示例

eg: package="com.neusoft.localsetting.red"

替换 红色(酷炫)主题的apk名为localsetting_red.apk

替换 金色(酷睿)主题的apk名为localsetting_gold.apk

替换 年轻色(酷潮)主题的apk名为 localsetting_young.apk
注意:其中red、gold、young是需要打合定义的,目的是为了在切换主题时,获取对应主题的包名

3、每套主题如果对应2套UI(日夜模式)大家正常在对应的主题中建-night文件夹即可。

本地测试:将生成的overlay apk签名 push 到vendor/overlay/red/xxxx.apk目录下; 然后调用切换主题即可。

二、实现主题切换流程

主题切换流程:

主要使用原生方法IOverlayManager,调用setEnabledExclusive方法切换主题。

1、开机遍历获取/vendor/overlay下的主题包

使用如下代码可以把不同主题的包名方法一个list中,等后续切换时直接使用。
注意:这个包名是之前和app打合的,必须所有的app都要按照这个要求规则命名apk。

java 复制代码
private List<String> mRed1List = new ArrayList<String>(); //存放红色主题包名list
private List<String> mYoungList = new ArrayList<String>(); //存放young主题包名list
private IOverlayManager mOverlayManager = IOverlayManager.Stub.asInterface(ServiceManager.getService("overlay"));


    public void getThemeOverlayList() {
        IOverlayManager mOverlayManager = IOverlayManager.Stub.asInterface(ServiceManager.getService("overlay"));
        mRed1List.clear();
        mYoungList.clear();
        try {
            mOverlayInfoMap = mOverlayManager.getAllOverlays(0);
            for (List<OverlayInfo> overlayInfos : mOverlayInfoMap.values()) {
                for (int j = 0; j < overlayInfos.size(); j++) {
                    String overlayPackageName = overlayInfos.get(j).packageName;
                    if (overlayPackageName.contains("red")) {
                        mRed1List.add(overlayPackageName);
                    } else if (overlayPackageName.contains("young")) {
                        mYoungList.add(overlayPackageName);
                    }
                }
            }
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }

2、切换时遍历list,调用原生方法mOverlayManager.setEnabledExclusive切换主题。

复制代码
mOverlayManager.setEnabledExclusive(mYoungList.get(i), true, 0);方法是把对应的主题包资源替换进去。
复制代码
mOverlayManager.setEnabled(mRed1List.get(i), false, 0);方法是把主题包禁用,从而显示显示源代码里面的资源。
java 复制代码
    private void changeThemeColor(int themeChange) {
        List<Boolean> changeThemeResult = new ArrayList<>();
        List<String> changeResultFail = new ArrayList<>();
        try {
            if (themeChange == SettingConfig.THEME_RED) {
                if (mRed1List.size() != 0) {
                    for (int i = 0; i < mRed1List.size(); i++) {
                        boolean isSuccess = mOverlayManager.setEnabledExclusive(mRed1List.get(i), true, 0);
                        if (!isSuccess) {
                            changeThemeResult.add(isSuccess);
                            changeResultFail.add(mRed1List.get(i));
                        }
                    }
                    Log.d(TAG, "---mRed1List.get(i):" + mRed1List.toString());
                }
            } else if (themeChange == SettingConfig.THEME_YOUNG) {
                if (mYoungList.size() != 0) {
                    for (int i = 0; i < mYoungList.size(); i++) {
                        boolean isSuccess = mOverlayManager.setEnabledExclusive(mYoungList.get(i), true, 0);
                        if (!isSuccess) {
                            changeThemeResult.add(isSuccess);
                            changeResultFail.add(mYoungList.get(i));
                        }
                    }
                    Log.d(TAG, "---mYoungList.get(i):" + mYoungList.toString());
                }
            } else if (themeChange == SettingConfig.THEME_GOLD) {
                if (mRed1List.size() != 0) {
                    for (int i = 0; i < mRed1List.size(); i++) {
                        boolean isSuccess = mOverlayManager.setEnabled(mRed1List.get(i), false, 0);
                        if (!isSuccess) {
                            changeThemeResult.add(isSuccess);
                            changeResultFail.add(mRed1List.get(i));
                        }
                    }
                    Log.d(TAG, "---mRed1List.get(i):" + mRed1List.toString());
                }
                if (mYoungList.size() != 0) {
                    for (int i = 0; i < mYoungList.size(); i++) {
                        boolean isSuccess = mOverlayManager.setEnabled(mYoungList.get(i), false, 0);
                        if (!isSuccess) {
                            changeThemeResult.add(isSuccess);
                            changeResultFail.add(mYoungList.get(i));
                        }
                    }
                    Log.d(TAG, "---mYoungList.get(i):" + mYoungList.toString());
                }
            }

            if (changeThemeResult != null && changeThemeResult.size() > 0
                    && changeResultFail != null && changeResultFail.size() > 0) {
                Log.d(TAG, "---changeThemeResult:" + changeThemeResult.toString());
                Log.d(TAG, "---changeResultFail:" + changeResultFail.toString());
            } else {
                Log.d(TAG, "---changeThemeResult   无");
                Log.d(TAG, "---changeResultFail    无");
            }

        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }

总结一下,留着以后用。

相关推荐
xiangpanf4 小时前
Laravel 10.x重磅升级:五大核心特性解析
android
robotx7 小时前
安卓线程相关
android
消失的旧时光-19437 小时前
Android 面试高频:JSON 文件、大数据存储与断电安全(从原理到工程实践)
android·面试·json
dalancon8 小时前
VSYNC 信号流程分析 (Android 14)
android
dalancon8 小时前
VSYNC 信号完整流程2
android
dalancon8 小时前
SurfaceFlinger 上帧后 releaseBuffer 完整流程分析
android
用户69371750013849 小时前
不卷AI速度,我卷自己的从容——北京程序员手记
android·前端·人工智能
程序员Android10 小时前
Android 刷新一帧流程trace拆解
android
墨狂之逸才10 小时前
解决 Android/Gradle 编译报错:Comparison method violates its general contract!
android
阿明的小蝴蝶11 小时前
记一次Gradle环境的编译问题与解决
android·前端·gradle