实现
添加依赖
compile 'com.android.support:appcompat-v7:25.1.0'
由于Support Library在23.2.0的版本中才添加了Theme.AppCompat.DayNight主题,所以依赖的版本必须是高于23.2.0的,并且,这个特性支持的最低SDK版本为14,所以,需要兼容Android 4.0的设备,是不能使用这个特性的,在API Level 14以下的设备会默认使用亮色主题。不过现在4.0以下的设备应该比较少了吧,毕竟微信的minSdkVersion都设置为14了。
准备资源
-
让我们自己的主题继承并应用DayNight主题。
<style name="AppTheme" parent="Theme.AppCompat.DayNight.DarkActionBar"> <item name="colorPrimary">@color/colorPrimary</item> <item name="colorPrimaryDark">@color/colorPrimaryDark</item> <item name="colorAccent">@color/colorAccent</item> <!--customize your theme here--> </style>
-
新建夜间模式资源文件夹:在
res
目录下新建values-night
文件夹,然后在此目录下新建colors.xml
文件在夜间模式下的应用的资源。当然也可以根据需要新建drawable-night
,layout-night
等后缀为-night
的夜间资源文件夹。
我的values
和values-night
目录下的colors.xml
的内容如下:<?xml version="1.0" encoding="utf-8"?> <!--values-colors.xml--> <resources> <color name="colorPrimary">#009688</color> <color name="colorPrimaryDark">#00796B</color> <color name="colorAccent">#009688</color> <color name="textColorPrimary">#616161</color> <color name="viewBackground">@android:color/white</color> </resources> <!--values-night-colors.xml--> <?xml version="1.0" encoding="utf-8"?> <resources> <color name="colorPrimary">#35464e</color> <color name="colorPrimaryDark">#212a2f</color> <color name="colorAccent">#212a2f</color> <color name="textColorPrimary">#616161</color> <color name="viewBackground">#212a2f</color> </resources>
-
使Activity继承自AppCompatActivity。
public class MainActivity extends AppCompatActivity { // code here @Override protected void onCreate(Bundle savedInstanceState) { } }
应用
静态应用
在Application的继承类下设置初始主题。
public class App extends Application {
@Override
public void onCreate() {
super.onCreate();
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES);
// other code here
}
这里AppCompatDelegate.setDefaultNightMode()
方法可以接受的参数值有4个:
- MODE_NIGHT_NO. Always use the day (light) theme(一直应用日间(light)主题).
- MODE_NIGHT_YES. Always use the night (dark) theme(一直使用夜间(dark)主题).
- MODE_NIGHT_AUTO. Changes between day/night based on the time of day(根据当前时间在day/night主题间切换).
- MODE_NIGHT_FOLLOW_SYSTEM(默认选项). This setting follows the system’s setting, which is essentially MODE_NIGHT_NO(跟随系统,通常为MODE_NIGHT_NO).
我们可以在任何时候调用这个方法,因为这个方法是静态的。但是这个值并不是一直存在的,每次在开启进程时需要重新设置。在上面的代码中,我是在onCreate()
方法中设置的,网上也有大神建议在Activity或者Application的static代码块中设置。如下所示:
public class App extends Application {
static {
AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES);
}
@Override
public void onCreate() {
super.onCreate();
// other code here
}
动态应用
虽然上面的静态应用的设置非常简单,但是这种方式的应用场景还是太少了。我们更多的还是需要动态的根据需要动态的切换。
-
检测当前主题模式
int mode = getResources().getConfiguration().uiMode & Configuration.UI_MODE_NIGHT_MASK;
-
设置主题
if(mode == Configuration.UI_MODE_NIGHT_YES) { AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_NO); } else if(mode == Configuration.UI_MODE_NIGHT_NO) { AppCompatDelegate.setDefaultNightMode(AppCompatDelegate.MODE_NIGHT_YES); } else { // blah blah } recreate();
在调用
recreate()
方法之前,还可以创建一些动画进行过渡。而且,众所周知,调用recreate()
需要对一些数据进行保存,例如fragment,CheckBox,RadioBox等。如下所示:public class MainFragment extends Fragment { @Override public void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); if (savedInstanceState != null) { FragmentManager manager = getChildFragmentManager(); doubanMomentFragment = (DoubanMomentFragment) manager.getFragment(savedInstanceState, "douban"); } else { doubanMomentFragment = DoubanMomentFragment.newInstance(); } } @Override public void onSaveInstanceState(Bundle outState) { super.onSaveInstanceState(outState); FragmentManager manager = getChildFragmentManager(); manager.putFragment(outState, "douban", doubanMomentFragment); }
我们也可以把主题的值存储到SharedPreference中,已便于应用在下一次启动时自动应用主题。
作者:TonnyL
链接:http://www.jianshu.com/p/dcfcfcbda7ac
來源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。