我实际上在我的应用程序中具有此功能,此外,我允许用户在运行时更改主题。由于从首选项中读取值需要一些时间,因此我通过保存缓存值的全局可访问函数获取主题 ID。
正如已经指出的 - 创建一些 Android 主题,使用 this guide。您的styles.xml 文件中至少有两个<style> 项目。例如:
<style name="Theme.App.Light" parent="@style/Theme.Light">...</style>
<style name="Theme.App.Dark" parent="@style/Theme">...</style>
现在,您必须将其中一种样式应用于您的活动。我在活动的onCreate 方法中这样做,在任何其他调用之前:
setTheme(MyApplication.getThemeId());
getThemeId是一个返回缓存主题ID的方法:
public static int getThemeId()
{
return themeId;
}
该字段正在被其他方法更新:
public static void reloadTheme()
{
themeSetting = PreferenceManager.getDefaultSharedPreferences(context).getString("defaultTheme", "0");
if(themeSetting.equals("0"))
themeId = R.style.Theme_Light;
else
themeId = R.style.Theme_Dark;
}
当首选项发生变化时(当然在启动时)会调用它。这两个方法位于MyApplication 类中,该类扩展了Application。偏好更改侦听器在本文末尾进行了描述,位于主活动类中。
最后一件非常重要的事情——在活动开始时应用主题。假设,您只能在首选项屏幕中更改主题,并且只有一种方法可以到达那里,即只有一个(主要)活动,当您退出首选项屏幕时,此活动不会重新启动 - 旧主题仍然是用过的。这是解决方法(重新启动您的主要活动):
@Override
protected void onResume() {
super.onResume();
if(schduledRestart)
{
schduledRestart = false;
Intent i = getBaseContext().getPackageManager().getLaunchIntentForPackage( getBaseContext().getPackageName() );
i.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);
startActivity(i);
}
}
scheduledRestart 是一个布尔变量,最初设置为 false。当这个监听器改变主题时,它设置为 true,这也会更新前面提到的缓存主题 ID:
private class ThemeListener implements OnSharedPreferenceChangeListener{
@Override
public void onSharedPreferenceChanged(SharedPreferences spref, String key) {
if(key.equals("defaultTheme") && !spref.getString(key, "0").equals(MyApplication.getThemeSetting()))
{
MyApplication.reloadTheme();
schduledRestart = true;
}
}
sp = PreferenceManager.getDefaultSharedPreferences(this);
listener = new ThemeListener();
sp.registerOnSharedPreferenceChangeListener(listener);
记住要持有对侦听器对象的引用,否则它将被垃圾回收(并且将停止工作)。