I'm writing a PreferenceActivity
with a PreferenceFragment
, and an InstantiationException
is being thrown when the device's orientation changes (i.e. landscape to portrait and vice versa). Also, if I tap 'Ok' on the dialog that pops up ("Unfortunately, YourApp has stopped."), the main activity relaunches, and the PreferenceActivity
works perfectly in the new orientation until it is changed again. I can't seem to find anything about this online.
Stack trace:
05-26 09:28:10.860: E/AndroidRuntime(3439): FATAL EXCEPTION: main
05-26 09:28:10.860: E/AndroidRuntime(3439): Process: com.example.myapp, PID: 3439
05-26 09:28:10.860: E/AndroidRuntime(3439): java.lang.RuntimeException: Unable to start activity ComponentInfo{com.example.myapp/com.example.myapp.SettingsActivity}: android.app.Fragment$InstantiationException: Unable to instantiate fragment com.devgregw.strikedistance_androidphoneandtablet.SettingsActivity$2: make sure class name exists, is public, and has an empty constructor that is public
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2195)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2245)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.ActivityThread.handleRelaunchActivity(ActivityThread.java:3738)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.ActivityThread.access$900(ActivityThread.java:135)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1202)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.os.Handler.dispatchMessage(Handler.java:102)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.os.Looper.loop(Looper.java:136)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.ActivityThread.main(ActivityThread.java:5027)
05-26 09:28:10.860: E/AndroidRuntime(3439): at java.lang.reflect.Method.invokeNative(Native Method)
05-26 09:28:10.860: E/AndroidRuntime(3439): at java.lang.reflect.Method.invoke(Method.java:515)
05-26 09:28:10.860: E/AndroidRuntime(3439): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:779)
05-26 09:28:10.860: E/AndroidRuntime(3439): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:595)
05-26 09:28:10.860: E/AndroidRuntime(3439): at de.robv.android.xposed.XposedBridge.main(XposedBridge.java:132)
05-26 09:28:10.860: E/AndroidRuntime(3439): at dalvik.system.NativeStart.main(Native Method)
05-26 09:28:10.860: E/AndroidRuntime(3439): Caused by: android.app.Fragment$InstantiationException: Unable to instantiate fragment com.example.myapp.SettingsActivity$2: make sure class name exists, is public, and has an empty constructor that is public
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.Fragment.instantiate(Fragment.java:601)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.FragmentState.instantiate(Fragment.java:98)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.FragmentManagerImpl.restoreAllState(FragmentManager.java:1759)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.Activity.onCreate(Activity.java:899)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.preference.PreferenceActivity.onCreate(PreferenceActivity.java:514)
05-26 09:28:10.860: E/AndroidRuntime(3439): at com.example.myapp.SettingsActivity.onCreate(SettingsActivity.java:23)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.Activity.performCreate(Activity.java:5231)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1087)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2159)
05-26 09:28:10.860: E/AndroidRuntime(3439): ... 13 more
05-26 09:28:10.860: E/AndroidRuntime(3439): Caused by: java.lang.InstantiationException: can't instantiate class com.example.myapp.SettingsActivity$2; no empty constructor
05-26 09:28:10.860: E/AndroidRuntime(3439): at java.lang.Class.newInstanceImpl(Native Method)
05-26 09:28:10.860: E/AndroidRuntime(3439): at java.lang.Class.newInstance(Class.java:1208)
05-26 09:28:10.860: E/AndroidRuntime(3439): at android.app.Fragment.instantiate(Fragment.java:590)
05-26 09:28:10.860: E/AndroidRuntime(3439): ... 21 more
SettingsActivity.java
:
package com.example.myapp;
import android.content.SharedPreferences;
import android.content.SharedPreferences.OnSharedPreferenceChangeListener;
import android.os.Bundle;
import android.preference.ListPreference;
import android.preference.Preference;
import android.preference.PreferenceActivity;
import android.preference.PreferenceFragment;
import android.preference.PreferenceGroup;
public class SettingsActivity extends PreferenceActivity {
public OnSharedPreferenceChangeListener changeListener;
public SharedPreferences preferences;
public PreferenceFragment fragment;
public SettingsActivity() {
}
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
changeListener = new OnSharedPreferenceChangeListener() {
@Override
public void onSharedPreferenceChanged(
SharedPreferences sharedPreferences, String key) {
Preference preference = fragment.findPreference(key);
if (preference instanceof ListPreference) {
ListPreference listPreference = (ListPreference) preference;
listPreference.setSummary(listPreference.getEntry());
}
}
};
fragment = new PreferenceFragment() {
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
addPreferencesFromResource(R.xml.preferences);
preferences = getPreferenceScreen().getSharedPreferences();
}
@Override
public void onResume() {
super.onResume();
for (int i = 0; i < getPreferenceScreen().getPreferenceCount(); ++i) {
Preference preference = getPreferenceScreen()
.getPreference(i);
if (preference instanceof PreferenceGroup) {
PreferenceGroup preferenceGroup = (PreferenceGroup) preference;
for (int j = 0; j < preferenceGroup
.getPreferenceCount(); ++j) {
updatePreference(preferenceGroup.getPreference(j));
}
} else {
updatePreference(preference);
}
}
}
public void updatePreference(Preference preference) {
if (preference instanceof ListPreference) {
ListPreference listPreference = (ListPreference) preference;
listPreference.setSummary(listPreference.getEntry());
}
}
};
getFragmentManager().beginTransaction()
.replace(android.R.id.content, fragment).commit();
}
@Override
protected void onResume() {
super.onResume();
preferences.registerOnSharedPreferenceChangeListener(changeListener);
}
@Override
protected void onPause() {
super.onPause();
preferences.unregisterOnSharedPreferenceChangeListener(changeListener);
}
}
Your Fragment
should have a constructor
Caused by: android.app.Fragment$InstantiationException: Unable to instantiate fragment com.example.myapp.SettingsActivity$2: make sure class name exists, is public, and has an empty constructor that is public
Also put the Fragment
declaration out of that function as a static inner class or in another .java file.
For more information, you can read the docs here. Or you can take a look at the Android training guide about Fragments here.