不闪屏切换Android App主题

最近想给自己的一个闹钟App增加一个夜间模式,一个比较简便的切换主题的方式就是在Styles.xml中设置两套Theme,分别是白天模式的主题Theme,还有一个是夜间模式的Theme。然后,通过在该Activity中的setContentView()方法之前,使用setTheme(...)方法设置Activity的Theme。但是重新设置的主题Theme必须调用recreate()方法使得Activity重新调用onCreate()方法才能表现出来。因此,这个时候会闪屏。我们先详细介绍下这种方式的实现,后面再说解决闪屏的方法。

第一步,确认哪些属性是需要根据主题变化而改变的

以下面这个活动为例,我希望在点击Change Theme按钮后,可以改变

  • StatusBar的颜色
  • ToolBar的颜色
  • 整个Activity的背景颜色
  • 以及“Hello Theme”这个TextView的背景颜色
    在以上打算随着主题Theme修改的属性中,前三个属性是可以直接在Theme的style.xml文件中设置的,最后一个TextView的背景颜色是需要首先自定义一个属性,然后才能够在style.xml文件中设置的
不闪屏切换Android App主题_第1张图片
3.JPG

第二步自定义属性

在values文件夹下,新建attrs.xml文件。下面的代码代表这新建一个名为"Text_bg_Color"的属性,该属性值是color类型或者reference引用类型。



    

第三步定义不同的主题风格

在styles.xml中定义不同Theme,以便后面进行切换。下面分别定义了两个Theme的style。第一个是默认的主题,也就是白天模式。第二个是夜间模式。两个模式都是继承了"Theme.AppCompat.Light.NoActionBar",所以绝大多属性都是一样的,不同的是分别自定义了一些属性。

  • colorPrimary是Toolbar的背景颜色
  • colorPrimaryDark是StatusBar的颜色
  • android:textColorPrimary是主标题的字体颜色
  • android:colorControlNormal是控制元件的默认状态颜色以及overflow menu(三个点)的颜色
  • colorAccent是控制元件在选中状态的颜色
  • android:windowBackground是Activity的背景颜色


    
    

    

在布局文件中,TextView要使用对应style中的TextView_bg_Color属性。"?attr/TextView_bg_Color"代表使用自定义属性TextView_bg_Color的值,而该属性的已经呗Theme文件所定义。所以,TextView的背景颜色就被Theme所定义了。

    

第四步在Activity中切换主题Theme

自定义一个ThemeUtile类,这是一个帮助切换主题的工具类。这个类提供了一个public static的布尔型的变量night,用来记录整个App所处于的主题模式。这个类还提供了一个public static方法changeTheme(),根据night的值,来对所有的Activity设置主题。

public class ThemeUtile {
    public static boolean night = false;
    public static void changeTheme(Activity activity){
        if (ThemeUtile.night){
            activity.setTheme(R.style.AppThemeNight);
        }else{
            activity.setTheme(R.style.AppTheme);
        }
    }
}

接着设置切换主题Button的点击事件。通过设置不同的night值,来设置不同的主题模式。因为setTheme()方法必须要在setContentView()方法之前调用,所以为了使当前Activity的主题切换成功,需要调用recreate()方法来重新调用onCreate()方法。这样也导致了当前Activity被销毁,并重新启动,所以会出现闪屏的现象。

        Button button = (Button) findViewById(R.id.Theme_Button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (ThemeUtile.night) {
                    ThemeUtile.night = false;
                } else {
                    ThemeUtile.night = true;
                }
                recreate();
            }
        });

夜间模式效果如下。


不闪屏切换Android App主题_第2张图片
night.JPG

第五步 解决闪屏的问题

我搜索了很多解决不闪屏切换主题的方法,要么效果不太好,要么比较“难”,需要较深的知识积累,我就想了一个比较取巧的方法,但是没有那么优雅。由于在当前屏幕值重新设置主题,会导致重新调用onCreate()方法导致闪屏。我们可以在一个新开的Activity中通过ThemeUtile.night变量重新设置主题,但不调用recreate()方法切换主题,而是“手动”设置需要改变的属性,在退出该Activity时,使用Intent回到之前的界面,并 intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK|Intent.FLAG_ACTIVITY_NEW_TASK);,使之前的Activity全部出栈,重新创建一个新的Activity,执行onCreate()方法,从而改变主题。
代码如下:

//在onCreate()方法中
ThemeUtile.changeTheme(this);
setContentView(R.layout.second_activity);
super.onCreate(savedInstanceState);
        ...
//设置改变主题按钮的点击事件
button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (! ThemeUtile.night) {
                    findViewById(R.id.sec_activity).setBackgroundColor(getResources().getColor(R.color.colorGray));
                    getWindow().setStatusBarColor(getResources().getColor(R.color.colorGray));
                    toolbar.setBackgroundColor(getResources().getColor(R.color.colorGray));
                    ThemeUtile.night = true;
                }else
                {
                    findViewById(R.id.sec_activity).setBackgroundColor(getResources().getColor(R.color.colorWhite));
                    getWindow().setStatusBarColor(getResources().getColor(R.color.colorLight));
                    toolbar.setBackgroundColor(getResources().getColor(R.color.colorLight));
                    ThemeUtile.night = false;
                }
            }
        });


//重写onBackPressed()方法
    @Override
    public void onBackPressed() {
        Intent intent = new Intent(SecondActivity.this,MainActivity.class);
        intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK|Intent.FLAG_ACTIVITY_NEW_TASK);
        startActivity(intent);
        super.onBackPressed();
    }

就是这样,虽然不是很优雅,但是完成了不闪屏切换Android App主题。

2016.06.03更新:
注意,setTheme()方法必须在setContentView()super.onCreate()之前调用,否则,Theme中的某些属性将无法显示出来。

你可能感兴趣的:(不闪屏切换Android App主题)