16904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler/* 26904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Copyright (C) 2015 The Android Open Source Project 36904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 46904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Licensed under the Apache License, Version 2.0 (the "License"); 56904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * you may not use this file except in compliance with the License. 66904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * You may obtain a copy of the License at 76904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 86904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * http://www.apache.org/licenses/LICENSE-2.0 96904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 106904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Unless required by applicable law or agreed to in writing, software 116904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * distributed under the License is distributed on an "AS IS" BASIS, 126904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 136904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * See the License for the specific language governing permissions and 146904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * limitations under the License 156904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 166904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 176904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerpackage android.support.v7.preference; 186904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 196904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.content.Context; 206904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.content.res.TypedArray; 211fcd004668b5a9038c312e4426031027d8641353Tony Mantlerimport android.graphics.Canvas; 221fcd004668b5a9038c312e4426031027d8641353Tony Mantlerimport android.graphics.Rect; 231fcd004668b5a9038c312e4426031027d8641353Tony Mantlerimport android.graphics.drawable.Drawable; 246904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.os.Bundle; 256904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.os.Handler; 266904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.os.Message; 276904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.support.annotation.Nullable; 28c39d9c75590eca86a5e7e32a8824ba04a0d42e9bAlan Viveretteimport android.support.annotation.RestrictTo; 296904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.support.annotation.XmlRes; 306904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.support.v4.app.DialogFragment; 316904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.support.v4.app.Fragment; 321fcd004668b5a9038c312e4426031027d8641353Tony Mantlerimport android.support.v4.view.ViewCompat; 3331453bcbebe270c126f9980f69d4626228c71f0dTony Mantlerimport android.support.v7.preference.internal.AbstractMultiSelectListPreference; 346904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.support.v7.widget.LinearLayoutManager; 356904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.support.v7.widget.RecyclerView; 366904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.util.TypedValue; 376904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.view.ContextThemeWrapper; 386904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.view.LayoutInflater; 396904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.view.View; 406904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerimport android.view.ViewGroup; 416904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 42c39d9c75590eca86a5e7e32a8824ba04a0d42e9bAlan Viveretteimport static android.support.annotation.RestrictTo.Scope.GROUP_ID; 43c39d9c75590eca86a5e7e32a8824ba04a0d42e9bAlan Viverette 446904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler/** 456904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Shows a hierarchy of {@link Preference} objects as 466904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * lists. These preferences will 476904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * automatically save to {@link android.content.SharedPreferences} as the user interacts with 486904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * them. To retrieve an instance of {@link android.content.SharedPreferences} that the 496904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * preference hierarchy in this fragment will use, call 506904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link PreferenceManager#getDefaultSharedPreferences(android.content.Context)} 516904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * with a context in the same package as this fragment. 526904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <p> 536904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Furthermore, the preferences shown will follow the visual style of system 546904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * preferences. It is easy to create a hierarchy of preferences (that can be 556904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * shown on multiple screens) via XML. For these reasons, it is recommended to 566904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * use this fragment (as a superclass) to deal with preferences in applications. 576904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <p> 586904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * A {@link PreferenceScreen} object should be at the top of the preference 596904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * hierarchy. Furthermore, subsequent {@link PreferenceScreen} in the hierarchy 606904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * denote a screen break--that is the preferences contained within subsequent 616904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link PreferenceScreen} should be shown on another screen. The preference 62ee1e0dfa158dab873977c50c746749f55a4c1251Tony Mantler * framework handles this by calling {@link #onNavigateToScreen(PreferenceScreen)}. 636904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <p> 646904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * The preference hierarchy can be formed in multiple ways: 656904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <li> From an XML file specifying the hierarchy 666904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <li> From different {@link android.app.Activity Activities} that each specify its own 676904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * preferences in an XML file via {@link android.app.Activity} meta-data 686904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <li> From an object hierarchy rooted with {@link PreferenceScreen} 696904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <p> 706904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * To inflate from XML, use the {@link #addPreferencesFromResource(int)}. The 716904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * root element should be a {@link PreferenceScreen}. Subsequent elements can point 726904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * to actual {@link Preference} subclasses. As mentioned above, subsequent 736904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link PreferenceScreen} in the hierarchy will result in the screen break. 746904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <p> 756904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * To specify an object hierarchy rooted with {@link PreferenceScreen}, use 766904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link #setPreferenceScreen(PreferenceScreen)}. 776904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <p> 786904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * As a convenience, this fragment implements a click listener for any 796904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * preference in the current hierarchy, see 806904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link #onPreferenceTreeClick(Preference)}. 816904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 826904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <div class="special reference"> 836904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <h3>Developer Guides</h3> 846904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <p>For information about using {@code PreferenceFragment}, 856904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * read the <a href="{@docRoot}guide/topics/ui/settings.html">Settings</a> 866904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * guide.</p> 876904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * </div> 886904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 896904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <a name="SampleCode"></a> 906904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <h3>Sample Code</h3> 916904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 926904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <p>The following sample code shows a simple preference fragment that is 936904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * populated from a resource. The resource it loads is:</p> 946904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 958bce9f50e2c7a6f0a84f843ae10e8eb275afc51aTony Mantler * {@sample frameworks/support/samples/SupportPreferenceDemos/res/xml/preferences.xml preferences} 966904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 976904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * <p>The fragment implementation itself simply populates the preferences 986904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * when created. Note that the preferences framework takes care of loading 996904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * the current values out of the app preferences and writing them when changed:</p> 1006904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 1018bce9f50e2c7a6f0a84f843ae10e8eb275afc51aTony Mantler * {@sample frameworks/support/samples/SupportPreferenceDemos/src/com/example/android/supportpreference/FragmentSupportPreferencesCompat.java 1028bce9f50e2c7a6f0a84f843ae10e8eb275afc51aTony Mantler * support_fragment_compat} 1036904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 1046904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @see Preference 1056904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @see PreferenceScreen 1066904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 1076904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantlerpublic abstract class PreferenceFragmentCompat extends Fragment implements 1086904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler PreferenceManager.OnPreferenceTreeClickListener, 1096904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler PreferenceManager.OnDisplayPreferenceDialogListener, 1106904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler PreferenceManager.OnNavigateToScreenListener, 1116904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler DialogPreference.TargetFragment { 1126904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 1136904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 1146904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Fragment argument used to specify the tag of the desired root 1156904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link android.support.v7.preference.PreferenceScreen} object. 1166904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 1176904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public static final String ARG_PREFERENCE_ROOT = 1186904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler "android.support.v7.preference.PreferenceFragmentCompat.PREFERENCE_ROOT"; 1196904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 1206904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private static final String PREFERENCES_TAG = "android:preferences"; 1216904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 1226904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private static final String DIALOG_FRAGMENT_TAG = 1236904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler "android.support.v7.preference.PreferenceFragment.DIALOG"; 1246904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 1256904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private PreferenceManager mPreferenceManager; 1266904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private RecyclerView mList; 1276904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private boolean mHavePrefs; 1286904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private boolean mInitDone; 1296904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 1306904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private Context mStyledContext; 1316904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 1326904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private int mLayoutResId = R.layout.preference_list_fragment; 1336904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 1341fcd004668b5a9038c312e4426031027d8641353Tony Mantler private final DividerDecoration mDividerDecoration = new DividerDecoration(); 1351fcd004668b5a9038c312e4426031027d8641353Tony Mantler 1366904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private static final int MSG_BIND_PREFERENCES = 1; 1376904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private Handler mHandler = new Handler() { 1386904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler @Override 1396904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void handleMessage(Message msg) { 1406904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler switch (msg.what) { 1416904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 1426904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler case MSG_BIND_PREFERENCES: 1436904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler bindPreferences(); 1446904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler break; 1456904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 1466904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 1476904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler }; 1486904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 1496904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final private Runnable mRequestFocus = new Runnable() { 150dc8e099ed130434c0238f558277c5bcb6a533121Aurimas Liutikas @Override 1516904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void run() { 1526904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mList.focusableViewAvailable(mList); 1536904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 1546904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler }; 1556904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 15666222008cbda61251014caf1442930a48561d25cTony Mantler private Runnable mSelectPreferenceRunnable; 15766222008cbda61251014caf1442930a48561d25cTony Mantler 1586904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 1596904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Interface that PreferenceFragment's containing activity should 1606904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * implement to be able to process preference items that wish to 1616904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * switch to a specified fragment. 1626904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 1636904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public interface OnPreferenceStartFragmentCallback { 1646904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 1656904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Called when the user has clicked on a Preference that has 1666904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * a fragment class name associated with it. The implementation 1676904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * should instantiate and switch to an instance of the given 1686904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * fragment. 169bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler * @param caller The fragment requesting navigation. 170bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler * @param pref The preference requesting the fragment. 171bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler * @return true if the fragment creation has been handled 1726904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 1736904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler boolean onPreferenceStartFragment(PreferenceFragmentCompat caller, Preference pref); 1746904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 1756904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 1766904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 1776904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Interface that PreferenceFragment's containing activity should 1786904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * implement to be able to process preference items that wish to 1796904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * switch to a new screen of preferences. 1806904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 1816904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public interface OnPreferenceStartScreenCallback { 1826904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 1836904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Called when the user has clicked on a PreferenceScreen item in order to navigate to a new 1846904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * screen of preferences. 1856904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param caller The fragment requesting navigation. 1866904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param pref The preference screen to navigate to. 187bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler * @return true if the screen navigation has been handled 1886904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 1896904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler boolean onPreferenceStartScreen(PreferenceFragmentCompat caller, PreferenceScreen pref); 1906904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 1916904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 192bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler public interface OnPreferenceDisplayDialogCallback { 193bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler 194bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler /** 195bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler * 196bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler * @param caller The fragment containing the preference requesting the dialog. 197bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler * @param pref The preference requesting the dialog. 198bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler * @return true if the dialog creation has been handled. 199bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler */ 200bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler boolean onPreferenceDisplayDialog(PreferenceFragmentCompat caller, Preference pref); 201bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler } 202bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler 2036904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler @Override 2046904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void onCreate(Bundle savedInstanceState) { 2056904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler super.onCreate(savedInstanceState); 2066904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final TypedValue tv = new TypedValue(); 2076904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler getActivity().getTheme().resolveAttribute(R.attr.preferenceTheme, tv, true); 2086904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final int theme = tv.resourceId; 2096904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (theme <= 0) { 2106904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler throw new IllegalStateException("Must specify preferenceTheme in theme"); 2116904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 2126904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mStyledContext = new ContextThemeWrapper(getActivity(), theme); 2136904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 2146904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mPreferenceManager = new PreferenceManager(mStyledContext); 2156904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mPreferenceManager.setOnNavigateToScreenListener(this); 2166904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final Bundle args = getArguments(); 2176904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final String rootKey; 2186904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (args != null) { 2196904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler rootKey = getArguments().getString(ARG_PREFERENCE_ROOT); 2206904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } else { 2216904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler rootKey = null; 2226904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 2236904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler onCreatePreferences(savedInstanceState, rootKey); 2246904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 2256904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 2266904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 2276904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Called during {@link #onCreate(Bundle)} to supply the preferences for this fragment. 2286904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Subclasses are expected to call {@link #setPreferenceScreen(PreferenceScreen)} either 2296904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * directly or via helper methods such as {@link #addPreferencesFromResource(int)}. 2306904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 2316904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param savedInstanceState If the fragment is being re-created from 2326904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * a previous saved state, this is the state. 2336904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param rootKey If non-null, this preference fragment should be rooted at the 2346904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link android.support.v7.preference.PreferenceScreen} with this key. 2356904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 2366904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public abstract void onCreatePreferences(Bundle savedInstanceState, String rootKey); 2376904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 2386904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler @Override 2396904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public View onCreateView(LayoutInflater inflater, ViewGroup container, 2406904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler Bundle savedInstanceState) { 2416904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 2426904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler TypedArray a = mStyledContext.obtainStyledAttributes(null, 2436904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler R.styleable.PreferenceFragmentCompat, 2441fcd004668b5a9038c312e4426031027d8641353Tony Mantler R.attr.preferenceFragmentCompatStyle, 2456904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 0); 2466904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 2471fcd004668b5a9038c312e4426031027d8641353Tony Mantler mLayoutResId = a.getResourceId(R.styleable.PreferenceFragmentCompat_android_layout, 2486904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mLayoutResId); 2496904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 2501fcd004668b5a9038c312e4426031027d8641353Tony Mantler final Drawable divider = a.getDrawable( 2511fcd004668b5a9038c312e4426031027d8641353Tony Mantler R.styleable.PreferenceFragmentCompat_android_divider); 252301061d5655864f03f9bab0a28f7c3f2e28a0a04Tony Mantler final int dividerHeight = a.getDimensionPixelSize( 2531fcd004668b5a9038c312e4426031027d8641353Tony Mantler R.styleable.PreferenceFragmentCompat_android_dividerHeight, -1); 2541fcd004668b5a9038c312e4426031027d8641353Tony Mantler 2556904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler a.recycle(); 2566904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 25726442f8503691eccb80169a47ecdcc4e862f4dcfTony Mantler // Need to theme the inflater to pick up the preferenceFragmentListStyle 25826442f8503691eccb80169a47ecdcc4e862f4dcfTony Mantler final TypedValue tv = new TypedValue(); 25926442f8503691eccb80169a47ecdcc4e862f4dcfTony Mantler getActivity().getTheme().resolveAttribute(R.attr.preferenceTheme, tv, true); 26026442f8503691eccb80169a47ecdcc4e862f4dcfTony Mantler final int theme = tv.resourceId; 26126442f8503691eccb80169a47ecdcc4e862f4dcfTony Mantler 26226442f8503691eccb80169a47ecdcc4e862f4dcfTony Mantler final Context themedContext = new ContextThemeWrapper(inflater.getContext(), theme); 26326442f8503691eccb80169a47ecdcc4e862f4dcfTony Mantler final LayoutInflater themedInflater = inflater.cloneInContext(themedContext); 26426442f8503691eccb80169a47ecdcc4e862f4dcfTony Mantler 26526442f8503691eccb80169a47ecdcc4e862f4dcfTony Mantler final View view = themedInflater.inflate(mLayoutResId, container, false); 2666904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 2673fadd62b614e4a69aefe920aac640bdb629e502eJason Monk final View rawListContainer = view.findViewById(AndroidResources.ANDROID_R_LIST_CONTAINER); 2686904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (!(rawListContainer instanceof ViewGroup)) { 2693fadd62b614e4a69aefe920aac640bdb629e502eJason Monk throw new RuntimeException("Content has view with id attribute " 2703fadd62b614e4a69aefe920aac640bdb629e502eJason Monk + "'android.R.id.list_container' that is not a ViewGroup class"); 2716904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 2726904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 2736904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final ViewGroup listContainer = (ViewGroup) rawListContainer; 2746904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 27526442f8503691eccb80169a47ecdcc4e862f4dcfTony Mantler final RecyclerView listView = onCreateRecyclerView(themedInflater, listContainer, 2766904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler savedInstanceState); 2776904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (listView == null) { 2786904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler throw new RuntimeException("Could not create RecyclerView"); 2796904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 2806904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 2816904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mList = listView; 2821fcd004668b5a9038c312e4426031027d8641353Tony Mantler 2831fcd004668b5a9038c312e4426031027d8641353Tony Mantler listView.addItemDecoration(mDividerDecoration); 2841fcd004668b5a9038c312e4426031027d8641353Tony Mantler setDivider(divider); 2851fcd004668b5a9038c312e4426031027d8641353Tony Mantler if (dividerHeight != -1) { 2861fcd004668b5a9038c312e4426031027d8641353Tony Mantler setDividerHeight(dividerHeight); 2871fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 2881fcd004668b5a9038c312e4426031027d8641353Tony Mantler 2896904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler listContainer.addView(mList); 2906904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mHandler.post(mRequestFocus); 29166222008cbda61251014caf1442930a48561d25cTony Mantler 2926904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler return view; 2936904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 2946904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 2951fcd004668b5a9038c312e4426031027d8641353Tony Mantler /** 2961fcd004668b5a9038c312e4426031027d8641353Tony Mantler * Sets the drawable that will be drawn between each item in the list. 2971fcd004668b5a9038c312e4426031027d8641353Tony Mantler * <p> 2981fcd004668b5a9038c312e4426031027d8641353Tony Mantler * <strong>Note:</strong> If the drawable does not have an intrinsic 2991fcd004668b5a9038c312e4426031027d8641353Tony Mantler * height, you should also call {@link #setDividerHeight(int)}. 3001fcd004668b5a9038c312e4426031027d8641353Tony Mantler * 3011fcd004668b5a9038c312e4426031027d8641353Tony Mantler * @param divider the drawable to use 3021fcd004668b5a9038c312e4426031027d8641353Tony Mantler * @attr ref R.styleable#PreferenceFragmentCompat_android_divider 3031fcd004668b5a9038c312e4426031027d8641353Tony Mantler */ 3041fcd004668b5a9038c312e4426031027d8641353Tony Mantler public void setDivider(Drawable divider) { 3051fcd004668b5a9038c312e4426031027d8641353Tony Mantler mDividerDecoration.setDivider(divider); 3061fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 3071fcd004668b5a9038c312e4426031027d8641353Tony Mantler 3081fcd004668b5a9038c312e4426031027d8641353Tony Mantler /** 3091fcd004668b5a9038c312e4426031027d8641353Tony Mantler * Sets the height of the divider that will be drawn between each item in the list. Calling 3101fcd004668b5a9038c312e4426031027d8641353Tony Mantler * this will override the intrinsic height as set by {@link #setDivider(Drawable)} 3111fcd004668b5a9038c312e4426031027d8641353Tony Mantler * 3121fcd004668b5a9038c312e4426031027d8641353Tony Mantler * @param height The new height of the divider in pixels. 3131fcd004668b5a9038c312e4426031027d8641353Tony Mantler * @attr ref R.styleable#PreferenceFragmentCompat_android_dividerHeight 3141fcd004668b5a9038c312e4426031027d8641353Tony Mantler */ 3151fcd004668b5a9038c312e4426031027d8641353Tony Mantler public void setDividerHeight(int height) { 3161fcd004668b5a9038c312e4426031027d8641353Tony Mantler mDividerDecoration.setDividerHeight(height); 3171fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 3181fcd004668b5a9038c312e4426031027d8641353Tony Mantler 3196904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler @Override 3208f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk public void onViewCreated(View view, @Nullable Bundle savedInstanceState) { 3218f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk super.onViewCreated(view, savedInstanceState); 3226904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 3236904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (mHavePrefs) { 3246904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler bindPreferences(); 32566222008cbda61251014caf1442930a48561d25cTony Mantler if (mSelectPreferenceRunnable != null) { 32666222008cbda61251014caf1442930a48561d25cTony Mantler mSelectPreferenceRunnable.run(); 32766222008cbda61251014caf1442930a48561d25cTony Mantler mSelectPreferenceRunnable = null; 32866222008cbda61251014caf1442930a48561d25cTony Mantler } 3296904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3306904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 3316904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mInitDone = true; 3328f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk } 3338f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk 3348f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk @Override 3358f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk public void onActivityCreated(Bundle savedInstanceState) { 3368f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk super.onActivityCreated(savedInstanceState); 3376904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 3386904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (savedInstanceState != null) { 3396904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler Bundle container = savedInstanceState.getBundle(PREFERENCES_TAG); 3406904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (container != null) { 3416904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final PreferenceScreen preferenceScreen = getPreferenceScreen(); 3426904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (preferenceScreen != null) { 3436904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler preferenceScreen.restoreHierarchyState(container); 3446904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3456904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3466904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3476904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3486904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 3496904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler @Override 3506904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void onStart() { 3516904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler super.onStart(); 3526904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mPreferenceManager.setOnPreferenceTreeClickListener(this); 3536904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mPreferenceManager.setOnDisplayPreferenceDialogListener(this); 3546904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3556904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 3566904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler @Override 3576904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void onStop() { 3586904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler super.onStop(); 3596904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mPreferenceManager.setOnPreferenceTreeClickListener(null); 3606904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mPreferenceManager.setOnDisplayPreferenceDialogListener(null); 3616904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3626904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 3636904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler @Override 3646904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void onDestroyView() { 3656904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mHandler.removeCallbacks(mRequestFocus); 3666904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mHandler.removeMessages(MSG_BIND_PREFERENCES); 3678f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk if (mHavePrefs) { 3688f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk unbindPreferences(); 3698f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk } 3708f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk mList = null; 3716904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler super.onDestroyView(); 3726904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3736904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 3746904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler @Override 3756904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void onSaveInstanceState(Bundle outState) { 3766904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler super.onSaveInstanceState(outState); 3776904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 3786904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final PreferenceScreen preferenceScreen = getPreferenceScreen(); 3796904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (preferenceScreen != null) { 3806904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler Bundle container = new Bundle(); 3816904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler preferenceScreen.saveHierarchyState(container); 3826904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler outState.putBundle(PREFERENCES_TAG, container); 3836904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3846904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3856904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 3866904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 3876904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Returns the {@link PreferenceManager} used by this fragment. 3886904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @return The {@link PreferenceManager}. 3896904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 3906904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public PreferenceManager getPreferenceManager() { 3916904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler return mPreferenceManager; 3926904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 3936904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 3946904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 3956904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Sets the root of the preference hierarchy that this fragment is showing. 3966904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 3976904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param preferenceScreen The root {@link PreferenceScreen} of the preference hierarchy. 3986904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 3996904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void setPreferenceScreen(PreferenceScreen preferenceScreen) { 4006904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (mPreferenceManager.setPreferences(preferenceScreen) && preferenceScreen != null) { 4016904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler onUnbindPreferences(); 4026904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mHavePrefs = true; 4036904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (mInitDone) { 4046904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler postBindPreferences(); 4056904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 4066904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 4076904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 4086904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 4096904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 4106904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Gets the root of the preference hierarchy that this fragment is showing. 4116904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 4126904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @return The {@link PreferenceScreen} that is the root of the preference 4136904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * hierarchy. 4146904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 4156904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public PreferenceScreen getPreferenceScreen() { 4166904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler return mPreferenceManager.getPreferenceScreen(); 4176904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 4186904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 4196904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 4206904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Inflates the given XML resource and adds the preference hierarchy to the current 4216904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * preference hierarchy. 4226904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 4236904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param preferencesResId The XML resource ID to inflate. 4246904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 4256904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void addPreferencesFromResource(@XmlRes int preferencesResId) { 4266904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler requirePreferenceManager(); 4276904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 4286904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler setPreferenceScreen(mPreferenceManager.inflateFromResource(mStyledContext, 4296904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler preferencesResId, getPreferenceScreen())); 4306904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 4316904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 4326904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 4336904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Inflates the given XML resource and replaces the current preference hierarchy (if any) with 4346904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * the preference hierarchy rooted at {@code key}. 4356904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 4366904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param preferencesResId The XML resource ID to inflate. 4376904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param key The preference key of the {@link android.support.v7.preference.PreferenceScreen} 4386904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * to use as the root of the preference hierarchy, or null to use the root 4396904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link android.support.v7.preference.PreferenceScreen}. 4406904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 4416904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void setPreferencesFromResource(@XmlRes int preferencesResId, @Nullable String key) { 4426904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler requirePreferenceManager(); 4436904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 4446904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final PreferenceScreen xmlRoot = mPreferenceManager.inflateFromResource(mStyledContext, 4456904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler preferencesResId, null); 4466904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 4476904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final Preference root; 4486904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (key != null) { 4496904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler root = xmlRoot.findPreference(key); 4506904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (!(root instanceof PreferenceScreen)) { 4516904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler throw new IllegalArgumentException("Preference object with key " + key 4526904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler + " is not a PreferenceScreen"); 4536904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 4546904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } else { 4556904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler root = xmlRoot; 4566904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 4576904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 4586904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler setPreferenceScreen((PreferenceScreen) root); 4596904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 4606904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 4616904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 4626904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@inheritDoc} 4636904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 464dc8e099ed130434c0238f558277c5bcb6a533121Aurimas Liutikas @Override 4656904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public boolean onPreferenceTreeClick(Preference preference) { 466c4868c8a2946a038466f51875a590759092137b2Tony Mantler if (preference.getFragment() != null) { 467c4868c8a2946a038466f51875a590759092137b2Tony Mantler boolean handled = false; 468b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler if (getCallbackFragment() instanceof OnPreferenceStartFragmentCallback) { 469b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler handled = ((OnPreferenceStartFragmentCallback) getCallbackFragment()) 470c4868c8a2946a038466f51875a590759092137b2Tony Mantler .onPreferenceStartFragment(this, preference); 471c4868c8a2946a038466f51875a590759092137b2Tony Mantler } 472c4868c8a2946a038466f51875a590759092137b2Tony Mantler if (!handled && getActivity() instanceof OnPreferenceStartFragmentCallback){ 473c4868c8a2946a038466f51875a590759092137b2Tony Mantler handled = ((OnPreferenceStartFragmentCallback) getActivity()) 474c4868c8a2946a038466f51875a590759092137b2Tony Mantler .onPreferenceStartFragment(this, preference); 475c4868c8a2946a038466f51875a590759092137b2Tony Mantler } 476c4868c8a2946a038466f51875a590759092137b2Tony Mantler return handled; 4776904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 4786904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler return false; 4796904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 4806904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 4816904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 4826904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Called by 4836904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link android.support.v7.preference.PreferenceScreen#onClick()} in order to navigate to a 4846904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * new screen of preferences. Calls 4856904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link PreferenceFragmentCompat.OnPreferenceStartScreenCallback#onPreferenceStartScreen} 486c4868c8a2946a038466f51875a590759092137b2Tony Mantler * if the target fragment or containing activity implements 4876904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link PreferenceFragmentCompat.OnPreferenceStartScreenCallback}. 4886904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param preferenceScreen The {@link android.support.v7.preference.PreferenceScreen} to 4896904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * navigate to. 4906904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 4916904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler @Override 4926904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void onNavigateToScreen(PreferenceScreen preferenceScreen) { 493bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler boolean handled = false; 494b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler if (getCallbackFragment() instanceof OnPreferenceStartScreenCallback) { 495b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler handled = ((OnPreferenceStartScreenCallback) getCallbackFragment()) 496bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler .onPreferenceStartScreen(this, preferenceScreen); 497bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler } 498bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler if (!handled && getActivity() instanceof OnPreferenceStartScreenCallback) { 499bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler ((OnPreferenceStartScreenCallback) getActivity()) 500bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler .onPreferenceStartScreen(this, preferenceScreen); 5016904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5026904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5036904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5046904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 5056904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Finds a {@link Preference} based on its key. 5066904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 5076904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param key The key of the preference to retrieve. 5086904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @return The {@link Preference} with the key, or null. 5096904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @see android.support.v7.preference.PreferenceGroup#findPreference(CharSequence) 5106904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 511dc8e099ed130434c0238f558277c5bcb6a533121Aurimas Liutikas @Override 5126904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public Preference findPreference(CharSequence key) { 5136904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (mPreferenceManager == null) { 5146904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler return null; 5156904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5166904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler return mPreferenceManager.findPreference(key); 5176904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5186904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5196904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private void requirePreferenceManager() { 5206904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (mPreferenceManager == null) { 5216904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler throw new RuntimeException("This should be called after super.onCreate."); 5226904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5236904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5246904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5256904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private void postBindPreferences() { 5266904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (mHandler.hasMessages(MSG_BIND_PREFERENCES)) return; 5276904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler mHandler.obtainMessage(MSG_BIND_PREFERENCES).sendToTarget(); 5286904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5296904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5306904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler private void bindPreferences() { 5316904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final PreferenceScreen preferenceScreen = getPreferenceScreen(); 5326904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (preferenceScreen != null) { 53353b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler getListView().setAdapter(onCreateAdapter(preferenceScreen)); 534e61b4475e1837034d8926593aff9d35f8dfaebe2Tony Mantler preferenceScreen.onAttached(); 5356904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5366904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler onBindPreferences(); 5376904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5386904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5398f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk private void unbindPreferences() { 5408f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk final PreferenceScreen preferenceScreen = getPreferenceScreen(); 5418f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk if (preferenceScreen != null) { 5428f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk preferenceScreen.onDetached(); 5438f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk } 5448f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk onUnbindPreferences(); 5458f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk } 5468f7f0d48fac8f83e90a04ba67f9a03f93c6ed262Jason Monk 5476904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** @hide */ 548c39d9c75590eca86a5e7e32a8824ba04a0d42e9bAlan Viverette @RestrictTo(GROUP_ID) 5496904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler protected void onBindPreferences() { 5506904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5516904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5526904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** @hide */ 553c39d9c75590eca86a5e7e32a8824ba04a0d42e9bAlan Viverette @RestrictTo(GROUP_ID) 5546904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler protected void onUnbindPreferences() { 5556904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5566904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5576904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public final RecyclerView getListView() { 5586904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler return mList; 5596904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5606904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5616904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 5626904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Creates the {@link android.support.v7.widget.RecyclerView} used to display the preferences. 5636904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Subclasses may override this to return a customized 5646904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link android.support.v7.widget.RecyclerView}. 5656904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param inflater The LayoutInflater object that can be used to inflate the 5666904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link android.support.v7.widget.RecyclerView}. 5676904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param parent The parent {@link android.view.View} that the RecyclerView will be attached to. 5686904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * This method should not add the view itself, but this can be used to generate 5696904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * the LayoutParams of the view. 5706904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param savedInstanceState If non-null, this view is being re-constructed from a previous 5716904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * saved state as given here 5726904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @return A new RecyclerView object to be placed into the view hierarchy 5736904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 5746904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public RecyclerView onCreateRecyclerView(LayoutInflater inflater, ViewGroup parent, 5756904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler Bundle savedInstanceState) { 5766904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler RecyclerView recyclerView = (RecyclerView) inflater 5776904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler .inflate(R.layout.preference_recyclerview, parent, false); 5786904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5796904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler recyclerView.setLayoutManager(onCreateLayoutManager()); 5801553af7799b550e0821a5e579db0bccf1f0835b8Steven Dao recyclerView.setAccessibilityDelegateCompat( 5811553af7799b550e0821a5e579db0bccf1f0835b8Steven Dao new PreferenceRecyclerViewAccessibilityDelegate(recyclerView)); 5826904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5836904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler return recyclerView; 5846904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5856904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5866904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 5876904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Called from {@link #onCreateRecyclerView} to create the 5886904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link android.support.v7.widget.RecyclerView.LayoutManager} for the created 5896904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * {@link android.support.v7.widget.RecyclerView}. 5906904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @return A new {@link android.support.v7.widget.RecyclerView.LayoutManager} instance. 5916904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 5926904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public RecyclerView.LayoutManager onCreateLayoutManager() { 5936904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler return new LinearLayoutManager(getActivity()); 5946904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 5956904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 5966904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler /** 59753b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler * Creates the root adapter. 59853b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler * 59953b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler * @param preferenceScreen Preference screen object to create the adapter for. 60053b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler * @return An adapter that contains the preferences contained in this {@link PreferenceScreen}. 60153b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler */ 60253b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler protected RecyclerView.Adapter onCreateAdapter(PreferenceScreen preferenceScreen) { 60353b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler return new PreferenceGroupAdapter(preferenceScreen); 60453b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler } 60553b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler 60653b6dca0375b0b40d968f9e9d3dabda3e958fc41Tony Mantler /** 6076904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * Called when a preference in the tree requests to display a dialog. Subclasses should 6086904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * override this method to display custom dialogs or to handle dialogs for custom preference 6096904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * classes. 6106904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * 6116904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler * @param preference The Preference object requesting the dialog. 6126904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler */ 6136904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler @Override 6146904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler public void onDisplayPreferenceDialog(Preference preference) { 615bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler 616bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler boolean handled = false; 617b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler if (getCallbackFragment() instanceof OnPreferenceDisplayDialogCallback) { 618b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler handled = ((OnPreferenceDisplayDialogCallback) getCallbackFragment()) 619bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler .onPreferenceDisplayDialog(this, preference); 620bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler } 621bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler if (!handled && getActivity() instanceof OnPreferenceDisplayDialogCallback) { 622bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler handled = ((OnPreferenceDisplayDialogCallback) getActivity()) 623bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler .onPreferenceDisplayDialog(this, preference); 624bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler } 625bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler 626bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler if (handled) { 627bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler return; 628bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler } 629bd4cbab53c68c95f62407cf872b7155a520748a0Tony Mantler 6306904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler // check if dialog is already showing 6316904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (getFragmentManager().findFragmentByTag(DIALOG_FRAGMENT_TAG) != null) { 6326904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler return; 6336904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 6346904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 6356904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler final DialogFragment f; 6366904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler if (preference instanceof EditTextPreference) { 6376904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler f = EditTextPreferenceDialogFragmentCompat.newInstance(preference.getKey()); 6386904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } else if (preference instanceof ListPreference) { 6396904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler f = ListPreferenceDialogFragmentCompat.newInstance(preference.getKey()); 64031453bcbebe270c126f9980f69d4626228c71f0dTony Mantler } else if (preference instanceof AbstractMultiSelectListPreference) { 64131453bcbebe270c126f9980f69d4626228c71f0dTony Mantler f = MultiSelectListPreferenceDialogFragmentCompat.newInstance(preference.getKey()); 6426904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } else { 6436904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler throw new IllegalArgumentException("Tried to display dialog for unknown " + 6446904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler "preference type. Did you forget to override onDisplayPreferenceDialog()?"); 6456904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 6466904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler f.setTargetFragment(this, 0); 6476904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler f.show(getFragmentManager(), DIALOG_FRAGMENT_TAG); 6486904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler } 6496904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler 650b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler /** 651b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler * Basically a wrapper for getParentFragment which is v17+. Used by the leanback preference lib. 652b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler * @return Fragment to possibly use as a callback 653b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler * @hide 654b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler */ 655c39d9c75590eca86a5e7e32a8824ba04a0d42e9bAlan Viverette @RestrictTo(GROUP_ID) 656b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler public Fragment getCallbackFragment() { 657b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler return null; 658b1ef047adbe527a195dbccf11bf886a43509aa31Tony Mantler } 6591fcd004668b5a9038c312e4426031027d8641353Tony Mantler 66066222008cbda61251014caf1442930a48561d25cTony Mantler public void scrollToPreference(final String key) { 66166222008cbda61251014caf1442930a48561d25cTony Mantler scrollToPreferenceInternal(null, key); 66266222008cbda61251014caf1442930a48561d25cTony Mantler } 66366222008cbda61251014caf1442930a48561d25cTony Mantler 66466222008cbda61251014caf1442930a48561d25cTony Mantler public void scrollToPreference(final Preference preference) { 66566222008cbda61251014caf1442930a48561d25cTony Mantler scrollToPreferenceInternal(preference, null); 66666222008cbda61251014caf1442930a48561d25cTony Mantler } 66766222008cbda61251014caf1442930a48561d25cTony Mantler 66866222008cbda61251014caf1442930a48561d25cTony Mantler private void scrollToPreferenceInternal(final Preference preference, final String key) { 66966222008cbda61251014caf1442930a48561d25cTony Mantler final Runnable r = new Runnable() { 67066222008cbda61251014caf1442930a48561d25cTony Mantler @Override 67166222008cbda61251014caf1442930a48561d25cTony Mantler public void run() { 67266222008cbda61251014caf1442930a48561d25cTony Mantler final RecyclerView.Adapter adapter = mList.getAdapter(); 67366222008cbda61251014caf1442930a48561d25cTony Mantler if (!(adapter instanceof 67466222008cbda61251014caf1442930a48561d25cTony Mantler PreferenceGroup.PreferencePositionCallback)) { 67566222008cbda61251014caf1442930a48561d25cTony Mantler if (adapter != null) { 67666222008cbda61251014caf1442930a48561d25cTony Mantler throw new IllegalStateException("Adapter must implement " 67766222008cbda61251014caf1442930a48561d25cTony Mantler + "PreferencePositionCallback"); 67866222008cbda61251014caf1442930a48561d25cTony Mantler } else { 67966222008cbda61251014caf1442930a48561d25cTony Mantler // Adapter was set to null, so don't scroll I guess? 68066222008cbda61251014caf1442930a48561d25cTony Mantler return; 68166222008cbda61251014caf1442930a48561d25cTony Mantler } 68266222008cbda61251014caf1442930a48561d25cTony Mantler } 68366222008cbda61251014caf1442930a48561d25cTony Mantler final int position; 68466222008cbda61251014caf1442930a48561d25cTony Mantler if (preference != null) { 68566222008cbda61251014caf1442930a48561d25cTony Mantler position = ((PreferenceGroup.PreferencePositionCallback) adapter) 68666222008cbda61251014caf1442930a48561d25cTony Mantler .getPreferenceAdapterPosition(preference); 68766222008cbda61251014caf1442930a48561d25cTony Mantler } else { 68866222008cbda61251014caf1442930a48561d25cTony Mantler position = ((PreferenceGroup.PreferencePositionCallback) adapter) 68966222008cbda61251014caf1442930a48561d25cTony Mantler .getPreferenceAdapterPosition(key); 69066222008cbda61251014caf1442930a48561d25cTony Mantler } 69166222008cbda61251014caf1442930a48561d25cTony Mantler if (position != RecyclerView.NO_POSITION) { 69266222008cbda61251014caf1442930a48561d25cTony Mantler mList.scrollToPosition(position); 69366222008cbda61251014caf1442930a48561d25cTony Mantler } else { 69466222008cbda61251014caf1442930a48561d25cTony Mantler // Item not found, wait for an update and try again 69566222008cbda61251014caf1442930a48561d25cTony Mantler adapter.registerAdapterDataObserver( 69666222008cbda61251014caf1442930a48561d25cTony Mantler new ScrollToPreferenceObserver(adapter, mList, preference, key)); 69766222008cbda61251014caf1442930a48561d25cTony Mantler } 69866222008cbda61251014caf1442930a48561d25cTony Mantler } 69966222008cbda61251014caf1442930a48561d25cTony Mantler }; 70066222008cbda61251014caf1442930a48561d25cTony Mantler if (mList == null) { 70166222008cbda61251014caf1442930a48561d25cTony Mantler mSelectPreferenceRunnable = r; 70266222008cbda61251014caf1442930a48561d25cTony Mantler } else { 70366222008cbda61251014caf1442930a48561d25cTony Mantler r.run(); 70466222008cbda61251014caf1442930a48561d25cTony Mantler } 70566222008cbda61251014caf1442930a48561d25cTony Mantler } 70666222008cbda61251014caf1442930a48561d25cTony Mantler 70766222008cbda61251014caf1442930a48561d25cTony Mantler private static class ScrollToPreferenceObserver extends RecyclerView.AdapterDataObserver { 70866222008cbda61251014caf1442930a48561d25cTony Mantler private final RecyclerView.Adapter mAdapter; 70966222008cbda61251014caf1442930a48561d25cTony Mantler private final RecyclerView mList; 71066222008cbda61251014caf1442930a48561d25cTony Mantler private final Preference mPreference; 71166222008cbda61251014caf1442930a48561d25cTony Mantler private final String mKey; 71266222008cbda61251014caf1442930a48561d25cTony Mantler 71366222008cbda61251014caf1442930a48561d25cTony Mantler public ScrollToPreferenceObserver(RecyclerView.Adapter adapter, RecyclerView list, 71466222008cbda61251014caf1442930a48561d25cTony Mantler Preference preference, String key) { 71566222008cbda61251014caf1442930a48561d25cTony Mantler mAdapter = adapter; 71666222008cbda61251014caf1442930a48561d25cTony Mantler mList = list; 71766222008cbda61251014caf1442930a48561d25cTony Mantler mPreference = preference; 71866222008cbda61251014caf1442930a48561d25cTony Mantler mKey = key; 71966222008cbda61251014caf1442930a48561d25cTony Mantler } 72066222008cbda61251014caf1442930a48561d25cTony Mantler 72166222008cbda61251014caf1442930a48561d25cTony Mantler private void scrollToPreference() { 72266222008cbda61251014caf1442930a48561d25cTony Mantler mAdapter.unregisterAdapterDataObserver(this); 72366222008cbda61251014caf1442930a48561d25cTony Mantler final int position; 72466222008cbda61251014caf1442930a48561d25cTony Mantler if (mPreference != null) { 72566222008cbda61251014caf1442930a48561d25cTony Mantler position = ((PreferenceGroup.PreferencePositionCallback) mAdapter) 72666222008cbda61251014caf1442930a48561d25cTony Mantler .getPreferenceAdapterPosition(mPreference); 72766222008cbda61251014caf1442930a48561d25cTony Mantler } else { 72866222008cbda61251014caf1442930a48561d25cTony Mantler position = ((PreferenceGroup.PreferencePositionCallback) mAdapter) 72966222008cbda61251014caf1442930a48561d25cTony Mantler .getPreferenceAdapterPosition(mKey); 73066222008cbda61251014caf1442930a48561d25cTony Mantler } 73166222008cbda61251014caf1442930a48561d25cTony Mantler if (position != RecyclerView.NO_POSITION) { 73266222008cbda61251014caf1442930a48561d25cTony Mantler mList.scrollToPosition(position); 73366222008cbda61251014caf1442930a48561d25cTony Mantler } 73466222008cbda61251014caf1442930a48561d25cTony Mantler } 73566222008cbda61251014caf1442930a48561d25cTony Mantler 73666222008cbda61251014caf1442930a48561d25cTony Mantler @Override 73766222008cbda61251014caf1442930a48561d25cTony Mantler public void onChanged() { 73866222008cbda61251014caf1442930a48561d25cTony Mantler scrollToPreference(); 73966222008cbda61251014caf1442930a48561d25cTony Mantler } 74066222008cbda61251014caf1442930a48561d25cTony Mantler 74166222008cbda61251014caf1442930a48561d25cTony Mantler @Override 74266222008cbda61251014caf1442930a48561d25cTony Mantler public void onItemRangeChanged(int positionStart, int itemCount) { 74366222008cbda61251014caf1442930a48561d25cTony Mantler scrollToPreference(); 74466222008cbda61251014caf1442930a48561d25cTony Mantler } 74566222008cbda61251014caf1442930a48561d25cTony Mantler 74666222008cbda61251014caf1442930a48561d25cTony Mantler @Override 74766222008cbda61251014caf1442930a48561d25cTony Mantler public void onItemRangeChanged(int positionStart, int itemCount, Object payload) { 74866222008cbda61251014caf1442930a48561d25cTony Mantler scrollToPreference(); 74966222008cbda61251014caf1442930a48561d25cTony Mantler } 75066222008cbda61251014caf1442930a48561d25cTony Mantler 75166222008cbda61251014caf1442930a48561d25cTony Mantler @Override 75266222008cbda61251014caf1442930a48561d25cTony Mantler public void onItemRangeInserted(int positionStart, int itemCount) { 75366222008cbda61251014caf1442930a48561d25cTony Mantler scrollToPreference(); 75466222008cbda61251014caf1442930a48561d25cTony Mantler } 75566222008cbda61251014caf1442930a48561d25cTony Mantler 75666222008cbda61251014caf1442930a48561d25cTony Mantler @Override 75766222008cbda61251014caf1442930a48561d25cTony Mantler public void onItemRangeRemoved(int positionStart, int itemCount) { 75866222008cbda61251014caf1442930a48561d25cTony Mantler scrollToPreference(); 75966222008cbda61251014caf1442930a48561d25cTony Mantler } 76066222008cbda61251014caf1442930a48561d25cTony Mantler 76166222008cbda61251014caf1442930a48561d25cTony Mantler @Override 76266222008cbda61251014caf1442930a48561d25cTony Mantler public void onItemRangeMoved(int fromPosition, int toPosition, int itemCount) { 76366222008cbda61251014caf1442930a48561d25cTony Mantler scrollToPreference(); 76466222008cbda61251014caf1442930a48561d25cTony Mantler } 76566222008cbda61251014caf1442930a48561d25cTony Mantler } 76666222008cbda61251014caf1442930a48561d25cTony Mantler 7671fcd004668b5a9038c312e4426031027d8641353Tony Mantler private class DividerDecoration extends RecyclerView.ItemDecoration { 7681fcd004668b5a9038c312e4426031027d8641353Tony Mantler 7691fcd004668b5a9038c312e4426031027d8641353Tony Mantler private Drawable mDivider; 7701fcd004668b5a9038c312e4426031027d8641353Tony Mantler private int mDividerHeight; 7711fcd004668b5a9038c312e4426031027d8641353Tony Mantler 7721fcd004668b5a9038c312e4426031027d8641353Tony Mantler @Override 7731fcd004668b5a9038c312e4426031027d8641353Tony Mantler public void onDrawOver(Canvas c, RecyclerView parent, RecyclerView.State state) { 7741fcd004668b5a9038c312e4426031027d8641353Tony Mantler if (mDivider == null) { 7751fcd004668b5a9038c312e4426031027d8641353Tony Mantler return; 7761fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 7771fcd004668b5a9038c312e4426031027d8641353Tony Mantler final int childCount = parent.getChildCount(); 7781fcd004668b5a9038c312e4426031027d8641353Tony Mantler final int width = parent.getWidth(); 7791fcd004668b5a9038c312e4426031027d8641353Tony Mantler for (int childViewIndex = 0; childViewIndex < childCount; childViewIndex++) { 7801fcd004668b5a9038c312e4426031027d8641353Tony Mantler final View view = parent.getChildAt(childViewIndex); 7811fcd004668b5a9038c312e4426031027d8641353Tony Mantler if (shouldDrawDividerBelow(view, parent)) { 7821fcd004668b5a9038c312e4426031027d8641353Tony Mantler int top = (int) ViewCompat.getY(view) + view.getHeight(); 7831fcd004668b5a9038c312e4426031027d8641353Tony Mantler mDivider.setBounds(0, top, width, top + mDividerHeight); 7841fcd004668b5a9038c312e4426031027d8641353Tony Mantler mDivider.draw(c); 7851fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 7861fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 7871fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 7881fcd004668b5a9038c312e4426031027d8641353Tony Mantler 7891fcd004668b5a9038c312e4426031027d8641353Tony Mantler @Override 7901fcd004668b5a9038c312e4426031027d8641353Tony Mantler public void getItemOffsets(Rect outRect, View view, RecyclerView parent, 7911fcd004668b5a9038c312e4426031027d8641353Tony Mantler RecyclerView.State state) { 7921fcd004668b5a9038c312e4426031027d8641353Tony Mantler if (shouldDrawDividerBelow(view, parent)) { 7931fcd004668b5a9038c312e4426031027d8641353Tony Mantler outRect.bottom = mDividerHeight; 7941fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 7951fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 7961fcd004668b5a9038c312e4426031027d8641353Tony Mantler 7971fcd004668b5a9038c312e4426031027d8641353Tony Mantler private boolean shouldDrawDividerBelow(View view, RecyclerView parent) { 7980eec6efa9177e1f72f97953dad7121004896ce1cTony Mantler final RecyclerView.ViewHolder holder = parent.getChildViewHolder(view); 7990eec6efa9177e1f72f97953dad7121004896ce1cTony Mantler final boolean dividerAllowedBelow = holder instanceof PreferenceViewHolder 8000eec6efa9177e1f72f97953dad7121004896ce1cTony Mantler && ((PreferenceViewHolder) holder).isDividerAllowedBelow(); 8010eec6efa9177e1f72f97953dad7121004896ce1cTony Mantler if (!dividerAllowedBelow) { 8020eec6efa9177e1f72f97953dad7121004896ce1cTony Mantler return false; 8030eec6efa9177e1f72f97953dad7121004896ce1cTony Mantler } 8041fcd004668b5a9038c312e4426031027d8641353Tony Mantler boolean nextAllowed = true; 8051fcd004668b5a9038c312e4426031027d8641353Tony Mantler int index = parent.indexOfChild(view); 8061fcd004668b5a9038c312e4426031027d8641353Tony Mantler if (index < parent.getChildCount() - 1) { 8071fcd004668b5a9038c312e4426031027d8641353Tony Mantler final View nextView = parent.getChildAt(index + 1); 8080eec6efa9177e1f72f97953dad7121004896ce1cTony Mantler final RecyclerView.ViewHolder nextHolder = parent.getChildViewHolder(nextView); 8090eec6efa9177e1f72f97953dad7121004896ce1cTony Mantler nextAllowed = nextHolder instanceof PreferenceViewHolder 8100eec6efa9177e1f72f97953dad7121004896ce1cTony Mantler && ((PreferenceViewHolder) nextHolder).isDividerAllowedAbove(); 8111fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 8120eec6efa9177e1f72f97953dad7121004896ce1cTony Mantler return nextAllowed; 8131fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 8141fcd004668b5a9038c312e4426031027d8641353Tony Mantler 8151fcd004668b5a9038c312e4426031027d8641353Tony Mantler public void setDivider(Drawable divider) { 8161fcd004668b5a9038c312e4426031027d8641353Tony Mantler if (divider != null) { 8171fcd004668b5a9038c312e4426031027d8641353Tony Mantler mDividerHeight = divider.getIntrinsicHeight(); 8181fcd004668b5a9038c312e4426031027d8641353Tony Mantler } else { 8191fcd004668b5a9038c312e4426031027d8641353Tony Mantler mDividerHeight = 0; 8201fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 8211fcd004668b5a9038c312e4426031027d8641353Tony Mantler mDivider = divider; 8221fcd004668b5a9038c312e4426031027d8641353Tony Mantler mList.invalidateItemDecorations(); 8231fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 8241fcd004668b5a9038c312e4426031027d8641353Tony Mantler 8251fcd004668b5a9038c312e4426031027d8641353Tony Mantler public void setDividerHeight(int dividerHeight) { 8261fcd004668b5a9038c312e4426031027d8641353Tony Mantler mDividerHeight = dividerHeight; 8271fcd004668b5a9038c312e4426031027d8641353Tony Mantler mList.invalidateItemDecorations(); 8281fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 8291fcd004668b5a9038c312e4426031027d8641353Tony Mantler } 8306904f67c96a28a0e5966b4fb6d37a0ad5f136858Tony Mantler} 831