android - 如何使用新的 Android-X API 从当前的 PreferenceFragment 打开一个新的 PreferenceFragment?

标签 android android-fragments android-preferences preferencefragment androidx

背景

在以前版本的支持库中,我们可以使用标题来获得设置的主菜单屏幕,每个屏幕都会打开一个新的设置屏幕( fragment )。

问题

现在 header 消失了(如写的 here )一段时间,我认为它在 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.

问题是,我无法在新的 android-x API 上使用它们。

每个 fragment 都有自己的首选项 XML 树(在 onCreatePreferences 中使用 setPreferencesFromResource),但我提出的每个解决方案要么什么也没做,要么崩溃了。

用可视化的方式来说,这就是我想要实现的目标:

enter image description here

由于有多个子设置屏幕,将它们的所有首选项都放在主设置屏幕的一个 XML 文件中会非常困惑。

我尝试过的

我唯一成功的是使用 PreferenceScreen 来保存应该显示的子屏幕的首选项。

这是一个这样的工作代码(项目可用 here):

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)
        }
    }
}

但是,正如我所写,这不是我想要做的。我想要多个扩展 PreferenceFragmentCompat 的类,每个类都有自己的 XML 文件,这些文件将从主文件中打开。

以下是我尝试过(但失败了)的方法:

  1. PreferenceScreen 设置一个“android:fragment”,指向新的 fragment 类,类似于标题。这根本没有做任何事情。

  2. 使用普通首选项并为其设置点击监听器,这将执行原始代码中显示的 fragment 事务。这导致了崩溃,显示类似“具有键 screen_preference 的首选项对象不是 PreferenceScreen”之类的内容。

  3. 尝试避免使用 ARG_PREFERENCE_ROOT ,但发生了与 #2 相同的崩溃。

  4. 按照建议here ,我试图在函数 getCallbackFragment 中返回 this,但这根本没有帮助。

问题

是否可以让主要设置 fragment 只让用户导航到其他 fragment ,而没有属于它们的任何其他首选项(在 preferences.xml 内)?

如何?

最佳答案

仅供引用,如果您使用 Navigation drawer + androidx.appcompat,您可以:

1) 将每个 PreferenceScreen 子级拆分为与 preference.xml 文件一样多的文件: 即“Pref_general.xml”将是主要首选项,“pref_ServerSettings.xml”包含带有您的服务器设置的 PreferenceScreen 子项。 2)为每个preference.xml创建一个PreferenceFragmentCompat:

“PrefFragmentGeneral”

在您的 PrefFragmentGeneral.xml 文件中,为任何子 xml 添加一个 Preference 而不是如下所示的 PreferenceScreen:

<Preference
    android:key="pref_serverPref"
    android:summary="@string/settings_serverPrefSum"
    android:title="@string/settings_serverPrefTitle"
    />

“PrefFragmentServer”

2) 确保覆盖“onCreatePreferences”以根据您想要的 XML 文件设置首选项:

public class PrefFragmentGeneral extends PreferenceFragmentCompat {
    @Override
    public void onCreatePreferences(Bundle savedInstanceState, String rootKey) {
        setPreferencesFromResource(R.xml.Pref_general, rootKey);
        //find your preference(s) using the same key
        Preference serverPref=findPreference("pref_serverPref");
        if(serverPref!=null){
            //Assign the click listener to navigate to the fragment using the navigation controller
            serverPref.setOnPreferenceClickListener(preference -> {
                NavController navController = Navigation.findNavController(getActivity(), R.id.nav_host_fragment);
                navController.navigate(R.id.nav_PrefFragmentServer);
                return true;
            });
        }
    }
//and the PrefFragmentServer 
public class PrefFragmentServer extends PreferenceFragmentCompat {
    @Override
    public void onCreatePreferences(Bundle savedInstanceState, String rootKey) {
        setPreferencesFromResource(R.xml.pref_ServerSettings,rootKey);
     }
}

3) 在抽屉导航中注册所有 fragment :

Navigation drawer

现在享受吧!

优点:当您向后导航时,您会回到“常规”首选项,就像您回到 PreferenceActivity 子级一样! 并且您不会收到异常,告诉您该 fragment 不是 FragmentManager 的一部分。

关于android - 如何使用新的 Android-X API 从当前的 PreferenceFragment 打开一个新的 PreferenceFragment?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/52723661/

相关文章:

java - NoClassDefError com.sun.jersey.api.client.Client

Android - 自定义 EditText 前缀颜色

android - getDefaultSharedPreferences 和 getSharedPreferences 之间的区别

android - Android 共享偏好值的最大大小是多少?

Android自定义评分栏图片大小问题

java - android webView 中的 ScrollView 不显示键盘

android: 方法 Split 可以返回空值吗?

android - 在多项 Activity 中留下 slider 的最佳方法应该是什么

android - 在 ViewPager 内的 ListView 中使用 OnItemClickListener

android - SharedPreferences.Editor commit() 是否只写回所做的更改或整个首选项集?