/* * Copyright (C) 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License */ package android.support.v17.preference; import static android.support.annotation.RestrictTo.Scope.LIBRARY_GROUP; import android.app.Fragment; import android.app.FragmentTransaction; import android.os.Build; import android.os.Bundle; import android.support.annotation.NonNull; import android.support.annotation.Nullable; import android.support.annotation.RestrictTo; import android.support.v14.preference.MultiSelectListPreference; import android.support.v14.preference.PreferenceFragment; import android.support.v7.preference.ListPreference; import android.support.v7.preference.Preference; import android.support.v7.preference.PreferenceScreen; import android.view.KeyEvent; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.Space; /** * This fragment provides a container for displaying a {@link LeanbackPreferenceFragment} * *

The following sample code shows a simple leanback preference fragment that is * populated from a resource. The resource it loads is:

* * {@sample frameworks/support/samples/SupportPreferenceDemos/res/xml/preferences.xml preferences} * *

The sample implements * {@link PreferenceFragment.OnPreferenceStartFragmentCallback#onPreferenceStartFragment(PreferenceFragment, Preference)}, * {@link PreferenceFragment.OnPreferenceStartScreenCallback#onPreferenceStartScreen(PreferenceFragment, PreferenceScreen)}, * and {@link #onPreferenceStartInitialScreen()}:

* * {@sample frameworks/support/samples/SupportPreferenceDemos/src/com/example/android/supportpreference/FragmentSupportPreferencesLeanback.java * support_fragment_leanback} */ public abstract class LeanbackSettingsFragment extends Fragment implements PreferenceFragment.OnPreferenceStartFragmentCallback, PreferenceFragment.OnPreferenceStartScreenCallback, PreferenceFragment.OnPreferenceDisplayDialogCallback { private static final String PREFERENCE_FRAGMENT_TAG = "android.support.v17.preference.LeanbackSettingsFragment.PREFERENCE_FRAGMENT"; private final RootViewOnKeyListener mRootViewOnKeyListener = new RootViewOnKeyListener(); @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { final View v = inflater.inflate(R.layout.leanback_settings_fragment, container, false); return v; } @Override public void onViewCreated(View view, Bundle savedInstanceState) { super.onViewCreated(view, savedInstanceState); if (savedInstanceState == null) { onPreferenceStartInitialScreen(); } } @Override public void onResume() { super.onResume(); // Trap back button presses final LeanbackSettingsRootView rootView = (LeanbackSettingsRootView) getView(); if (rootView != null) { rootView.setOnBackKeyListener(mRootViewOnKeyListener); } } @Override public void onPause() { super.onPause(); final LeanbackSettingsRootView rootView = (LeanbackSettingsRootView) getView(); if (rootView != null) { rootView.setOnBackKeyListener(null); } } @Override public boolean onPreferenceDisplayDialog(@NonNull PreferenceFragment caller, Preference pref) { if (caller == null) { throw new IllegalArgumentException("Cannot display dialog for preference " + pref + ", Caller must not be null!"); } final Fragment f; if (pref instanceof ListPreference) { final ListPreference listPreference = (ListPreference) pref; f = LeanbackListPreferenceDialogFragment.newInstanceSingle(listPreference.getKey()); f.setTargetFragment(caller, 0); startPreferenceFragment(f); } else if (pref instanceof MultiSelectListPreference) { MultiSelectListPreference listPreference = (MultiSelectListPreference) pref; f = LeanbackListPreferenceDialogFragment.newInstanceMulti(listPreference.getKey()); f.setTargetFragment(caller, 0); startPreferenceFragment(f); } // TODO // else if (pref instanceof EditTextPreference) { // // } else { return false; } return true; } /** * Called to instantiate the initial {@link android.support.v14.preference.PreferenceFragment} * to be shown in this fragment. Implementations are expected to call * {@link #startPreferenceFragment(android.app.Fragment)}. */ public abstract void onPreferenceStartInitialScreen(); /** * Displays a preference fragment to the user. This method can also be used to display * list-style fragments on top of the stack of preference fragments. * * @param fragment Fragment instance to be added. */ public void startPreferenceFragment(@NonNull Fragment fragment) { final FragmentTransaction transaction = getChildFragmentManager().beginTransaction(); final Fragment prevFragment = getChildFragmentManager().findFragmentByTag(PREFERENCE_FRAGMENT_TAG); if (prevFragment != null) { transaction .addToBackStack(null) .replace(R.id.settings_preference_fragment_container, fragment, PREFERENCE_FRAGMENT_TAG); } else { transaction .add(R.id.settings_preference_fragment_container, fragment, PREFERENCE_FRAGMENT_TAG); } transaction.commit(); } /** * Displays a fragment to the user, temporarily replacing the contents of this fragment. * * @param fragment Fragment instance to be added. */ public void startImmersiveFragment(@NonNull Fragment fragment) { final FragmentTransaction transaction = getChildFragmentManager().beginTransaction(); final Fragment preferenceFragment = getChildFragmentManager().findFragmentByTag(PREFERENCE_FRAGMENT_TAG); if (preferenceFragment != null && !preferenceFragment.isHidden()) { if (Build.VERSION.SDK_INT < 23) { // b/22631964 transaction.add(R.id.settings_preference_fragment_container, new DummyFragment()); } transaction.remove(preferenceFragment); } transaction .add(R.id.settings_dialog_container, fragment) .addToBackStack(null) .commit(); } private class RootViewOnKeyListener implements View.OnKeyListener { @Override public boolean onKey(View v, int keyCode, KeyEvent event) { if (keyCode == KeyEvent.KEYCODE_BACK) { return getChildFragmentManager().popBackStackImmediate(); } else { return false; } } } /** * @hide */ @RestrictTo(LIBRARY_GROUP) public static class DummyFragment extends Fragment { @Override public @Nullable View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { final View v = new Space(inflater.getContext()); v.setVisibility(View.GONE); return v; } } }