- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
创建首选项 Activity 后,我注意到尽管调用了 onSharedPreferenceChanged
,但当我的复选框首选项被选中时,我的主要 Activity 并没有改变主题。有谁知道出了什么问题以及如何解决这个问题?
样式.xml
<!-- Base application theme. -->
<style name="AppTheme" parent="android:Theme.Material.Light.DarkActionBar">
<!--<item name="android:windowBackground">@color/colorLight</item>-->
</style>
<style name="MyDarkMaterialTheme" parent="android:Theme.Material">
<item name="android:windowBackground">@android:color/black</item>
</style>
<style name="MyLightMaterialTheme" parent="android:Theme.Material.Light.DarkActionBar">
<item name="android:windowBackground">@color/colorLight</item>
</style>
MainActivity 类
public class MainActivity extends Activity {
boolean themeState;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setTheme(R.style.MyDarkMaterialTheme);
setContentView(R.layout.activity_main);
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.menu_main, menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
switch (item.getItemId()) {
case R.id.action_settings:
Intent settingsIntent = new Intent(this, SettingsActivity.class);
startActivity(settingsIntent);
return true;
default:
return super.onOptionsItemSelected(item);
}
}
@Override
public void onResume(){
super.onResume();
loadPreferences();
displaySettings();
}
private void loadPreferences(){
SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(this);
themeState = sharedPreferences.getBoolean("pref_pref1", true);
}
public void displaySettings() {
if (themeState) {
setTheme(R.style.MyDarkMaterialTheme);
recreate();
} else {
setTheme(R.style.MyLightMaterialTheme);
recreate();
}
}
}
SettingsActivity 类
public class SettingsActivity extends Activity {
boolean themeState;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_settings);
if (savedInstanceState == null) {
Fragment preferenceFragment = new SettingsFragment();
FragmentTransaction ft = getFragmentManager().beginTransaction();
ft.add(R.id.pref_container, preferenceFragment);
ft.commit();
}
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
if (item.getItemId() == android.R.id.home) {
final Intent intent = getParentActivityIntent();
if(intent != null){
intent.addFlags(Intent.FLAG_ACTIVITY_NO_ANIMATION);
}
onBackPressed();
return true;
}
return super.onOptionsItemSelected(item);
}
@Override
public void onResume(){
super.onResume();
loadPreferences();
displaySettings();
}
private void loadPreferences(){
SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(this);
themeState = sharedPreferences.getBoolean("pref_pref1", true);
}
public void displaySettings() {
if (themeState) {
getApplication().setTheme(R.style.MyDarkMaterialTheme);
recreate();
} else {
getApplication().setTheme(R.style.MyLightMaterialTheme);
recreate();
}
}
}
SettingsFragment 类
public class SettingsFragment extends PreferenceFragment {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//Load the Preferences from the XML file
addPreferencesFromResource(R.xml.app_preferences);
}
@Override
public void onActivityCreated(Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
new SharedPreferences.OnSharedPreferenceChangeListener() {
@Override
public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key) {
// Settings activity or fragment should restart with changes applied
}
};
}
}
xml/app_preferences
<?xml version="1.0" encoding="utf-8"?>
<PreferenceScreen
xmlns:android="http://schemas.android.com/apk/res/android">
<CheckBoxPreference
android:key="pref_pref1"
android:title="@string/dark_theme"
android:defaultValue="false"
android:layout="@layout/preference_multiline"
/>
</PreferenceScreen>
Csongi77的建议
public class SettingsFragment extends PreferenceFragment implements Preference.OnPreferenceChangeListener {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//Load the Preferences from the XML file
addPreferencesFromResource(R.xml.app_preferences);
// Find appropriate preference
CheckBoxPreference mThemePreference =(CheckBoxPreference)getPreferenceManager().findPreference("pref_pref1");
// we have to set up listener in order for persisting change to new value
mThemePreference.setOnPreferenceChangeListener(this);
SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(mThemePreference.getContext());
Boolean value=sharedPreferences.getBoolean("pref_pref1",true);
onPreferenceChange(mThemePreference, value);
}
// overriding onPreferenceChange - if we return true, the preference will be persisted
@Override
public boolean onPreferenceChange(Preference preference, Object newValue) {
String preferenceKey = preference.getKey();
// we have to check the preference type and key, maybe later we have more preferences....
if(preference instanceof CheckBoxPreference){
if(preferenceKey.equals("pref_pref1")){
((CheckBoxPreference)preference).setChecked((Boolean)newValue);
// ... do other preference related stuff here - if necessary, for example setSummary, etc...
getActivity().setTheme(R.style.MyDarkMaterialTheme);
} else {
getActivity().setTheme(R.style.MyLightMaterialTheme);
}
}
return true;
}
}
逻辑猫
Process: com.companyname.appname, PID: 4505
java.lang.NullPointerException: Attempt to invoke interface method 'void com.companyname.appname.SettingsFragment$PreferenceXchangeListener.onXchange(java.lang.Boolean)' on a null object reference
at com.companyname.appname.SettingsFragment.onPreferenceChange(SettingsFragment.java:57)
at android.preference.Preference.callChangeListener(Preference.java:928)
at android.preference.TwoStatePreference.onClick(TwoStatePreference.java:64)
at android.preference.Preference.performClick(Preference.java:983)
at android.preference.PreferenceScreen.onItemClick(PreferenceScreen.java:214)
at android.widget.AdapterView.performItemClick(AdapterView.java:300)
at android.widget.AbsListView.performItemClick(AbsListView.java:1143)
at android.widget.AbsListView$PerformClick.run(AbsListView.java:3044)
at android.widget.AbsListView$3.run(AbsListView.java:3833)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:135)
at android.app.ActivityThread.main(ActivityThread.java:5221)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:899)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:694)
SettingsActivity 类
public class SettingsActivity extends Activity implements SettingsFragment.PreferenceXchangeListener {
private static final String TAG = SettingsActivity.class.getSimpleName();
// declaring initial value for applying appropriate Theme
private Boolean mCurrentValue;
@Override
protected void onCreate(Bundle savedInstanceState) {
// Checking which Theme should be used. IMPORTANT: applying Themes MUST called BEFORE super.onCreate() and setContentView!!!
Log.d(TAG, "onCreate:::: retrieving preferences");
SharedPreferences mSharedPreferences = PreferenceManager.getDefaultSharedPreferences(this);
mCurrentValue = mSharedPreferences.getBoolean("my_preference",false);
Log.d(TAG, "onCreate:::: my_preference and mCurrentValue=" + mCurrentValue);
if(mCurrentValue){
// we have to use simple setTheme() instead getApplication.setTheme()!!!
setTheme(R.style.DarkTheme);
Log.d(TAG, "onCreate:::: setTheme:DarkTheme");
} else {
setTheme(R.style.LightTheme);
Log.d(TAG, "onCreate:::: setTheme:LightTheme");
}
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_settings);
Fragment preferenceFragment = new SettingsFragment();
getFragmentManager().beginTransaction().add(R.id.preference_container, preferenceFragment).commit();
}
// callback method for changing preference. It's called only if "my_preference" has changed
@Override
public void onXchange(Boolean value) {
// if value differs from previous Theme, we recreate Activity
Log.d(TAG, "onXchange:::: \n has called");
if (value!=mCurrentValue) {
Log.d(TAG, "onXchange:::: \n new value!=oldValue");
mCurrentValue=value;
recreate();
}
}
}
SettingsFragment 类
public class SettingsFragment extends PreferenceFragment implements Preference.OnPreferenceChangeListener {
private static final String TAG = SettingsFragment.class.getSimpleName();
// declaring PreferenceXchangeListener
private PreferenceXchangeListener mPreferenceXchangeListener;
public SettingsFragment() {
}
// declaring PreferenceXchangeListener in order to communicate with Activities
public interface PreferenceXchangeListener {
void onXchange(Boolean value);
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//Load the Preferences from the XML file
addPreferencesFromResource(R.xml.app_preferences);
CheckBoxPreference mCheckBoxPreference = (CheckBoxPreference)findPreference("my_preference");
mCheckBoxPreference.setOnPreferenceChangeListener(this);
}
@Override
public void onAttach(Context context) {
super.onAttach(context);
// on Attch we assign parent Activity as PreferenceXchangeListener
try {
mPreferenceXchangeListener = (PreferenceXchangeListener) context;
} catch (ClassCastException e) {
Log.e(TAG, "onAttach::::: PreferenceXchangeListener must be set in parent Activity");
}
}
@Override
public boolean onPreferenceChange(Preference preference, Object newValue) {
String preferenceKey=preference.getKey();
// only my_preference is checked in this case. Later you may add another behaviour to another preference change
if(preferenceKey.equals("my_preference")){
((CheckBoxPreference)preference).setChecked((Boolean)newValue);
// executing parent Activity's callback with the new value
mPreferenceXchangeListener.onXchange((Boolean)newValue);
return true;
}
// ... check other preferences here
return false;
}
}
最佳答案
试试这个:
public class SettingsFragment extends PreferenceFragment implements Preference.OnPreferenceChangeListener {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//Load the Preferences from the XML file
addPreferencesFromResource(R.xml.app_preferences);
// Find appropriate preference
CheckBoxPreference mThemePreference =(CheckBoxPreference)getPreferenceManager().findPreference("pref_pref1");
// we have to set up listener in order for persisting change to new value
mThemePreference.setOnPreferenceChangeListener(this);
SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(mThemePreference.getContext());
Boolean value=sharedPreferences.getBoolean("pref_pref1",true);
onPreferenceChange(mThemePreference, value);
}
// overriding onPreferenceChange - if we return 'true', the preference will be persisted
@Override
public boolean onPreferenceChange(Preference preference, Object newValue) {
String preferenceKey = preference.getKey();
// we have to check the preference type and key, maybe later we have more preferences....
if(preference instanceof CheckBoxPreference){
if(preferenceKey.equals("pref_pref1")){
// Edited this line *******
((CheckBoxPreference)preference).setChecked((Boolean)newValue);
// ... do other preference related stuff here - if necessary, for example setSummary, etc...
}
}
return true;
}
}
简而言之:在您的 PreferenceFragment 中实现 OnPreferenceChange。当您覆盖 onPreferenceChane 时返回 true。在这种情况下,旧的首选项将被覆盖。希望对您有所帮助(如果是,请不要忘记接受我的回答)!此致,铯
P.S: 不要忘记在模拟器上卸载应用
关于java - 调用 onSharedPreferenceChanged 后应用主题没有改变,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52084069/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!