Background
On previous versions of support library, we could use headers in order to have a main-menu screen of settings, that each would open a new settings screen (fragment) .
The problem
Now headers are gone (as written here) for some time, and I think it became worse on android-x :
One thing you’ll note isn’t in here is preference headers and you’d be totally right. However, that doesn’t mean a single list of preferences need to span a 10” tablet screen. Instead, your Activity can implement OnPreferenceStartFragmentCallback (link) to handle preferences with an app:fragment attribute or OnPreferenceStartScreenCallback (link) to handle PreferenceScreen preferences. This allows you to construct a ‘header’ style PreferenceFragmentCompat in one pane and use those callbacks to replace a second pane without working in two separate types of XML files.
Thing is, I fail to use these on the new android-x API.
Each fragment has its own preferences XML tree (using setPreferencesFromResource
within onCreatePreferences
) , but each solution I've come up with has either done nothing, or crashed.
To put it in a visual way, this is what I'm trying to achieve :
Since there are multiple sub settings screens, it would be very messy to have all of the preferences of all of them be put in one XML file of the main settings screen.
What I've tried
Only thing I've succeeded, is to use the PreferenceScreen to hold the preferences of the sub-screen that's supposed to be shown.
Here's a working code (project available here) of such a thing :
preferences.xml
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android" android:title="Demo">
<PreferenceScreen
android:key="screen_preference" android:summary="Shows another screen of preferences"
android:title="Screen preferenc">
<CheckBoxPreference
android:key="next_screen_checkbox_preference"
android:summary="Preference that is on the next screen but same hierarchy"
android:title="Toggle preference"/>
</PreferenceScreen>
</PreferenceScreen>
MainActivity.kt
class MainActivity : AppCompatActivity(), PreferenceFragmentCompat.OnPreferenceStartScreenCallback {
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
supportActionBar!!.setDisplayHomeAsUpEnabled(true)
if (savedInstanceState == null)
supportFragmentManager.beginTransaction().replace(android.R.id.content, PrefsFragment()).commit()
}
override fun onPreferenceStartScreen(caller: PreferenceFragmentCompat, pref: PreferenceScreen): Boolean {
val f = PrefsFragment()
val args = Bundle(1)
args.putString(PreferenceFragmentCompat.ARG_PREFERENCE_ROOT, pref.key)
f.arguments = args
supportFragmentManager.beginTransaction().replace(android.R.id.content, f).addToBackStack(null).commit()
return true
}
class PrefsFragment : PreferenceFragmentCompat() {
override fun onCreatePreferences(savedInstanceState: Bundle?, rootKey: String?) {
setPreferencesFromResource(R.xml.preferences, rootKey)
}
}
}
But, as I wrote, this is not what I'm trying to do. I want to have multiple classes that extend PreferenceFragmentCompat, each with its own XML file, which will be opened from the main one.
Here are the things I've tried (and failed) :
Set a "android:fragment" for the
PreferenceScreen
, to point to the new fragments classes, similar to headers. This didn't do anything at all.Use a normal Preference and have click listener for it, that will do the fragment transaction as shown on the original code. This caused a crash that says something like "Preference object with key screen_preference is not a PreferenceScreen" .
Tried to avoid using ARG_PREFERENCE_ROOT , but had same crash as on #2 .
As suggested here, I tried to return
this
in functiongetCallbackFragment
, but this didn't help at all.
The question
Is it possible to have the main settings fragment just let the user to navigate to the other fragments, while not having any other preferences that belong to them (inside preferences.xml
) ?
How?
What you tried in 1) was the correct approach - but you should not use
<PreferenceScreen>
tags for this.Your XML resource should look like this instead:
Also, if you are using a version of Preference older than
androidx.preference:preference:1.1.0-alpha01
, you will need to implement onPreferenceStartFragment to handle the fragment transaction. (in 1.1.0 alpha01 this method has a default implementation, but you are still encouraged to use your own implementation to customize any animations / transitions)This should look something like:
For more information you can check out the Settings guide and the AndroidX Preference Sample
EDIT: a sample of the first solution, after updating, available here.
Here's how it can work (sample available here) :
MainActivity.kt
preferences.xml
preferences2.xml
gradle dependencies:
OK, I've found 2 possible, yet weird, solutions.
I still would like to know if there is an official way to do it, because both solutions are quite weird.
Solution 1
In the main settings preference XML file, for each sub
PreferenceScreen
, I put an emptyPreference
tag.preferences.xml
I pass null for the second argument of
setPreferencesFromResource
on the new sub-screen fragment.Here's the code (project available here) :
MainActivity.kt
Of course, this needs to be modified so that you will know which fragment to create and add...
Solution 2
I use a normal
Preference
instead of eachPreferenceScreen
, and for each of them I choose to add the fragment upon clicking (project available here) :preferences.xml
MainActivity.kt
EDIT: a tiny modification to the second solution can make it nicer:
preferences.xml
MainActivity.kt
Note that you need to add this to Proguard rules:
Another improvement to solution #2 is that it can go over all preferences by itself:
EDIT: seems the first solution was the correct one, but needed a change. Check the answer here. Full sample available here.