Home | History | Annotate | Download | only in preference
      1 /*
      2  * Copyright (C) 2010 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 package android.preference;
     18 
     19 import android.content.Context;
     20 import android.content.SharedPreferences;
     21 import android.content.res.TypedArray;
     22 import android.os.Parcel;
     23 import android.os.Parcelable;
     24 import android.util.AttributeSet;
     25 import android.view.View;
     26 import android.view.accessibility.AccessibilityEvent;
     27 import android.view.accessibility.AccessibilityManager;
     28 import android.widget.TextView;
     29 
     30 /**
     31  * Common base class for preferences that have two selectable states, persist a
     32  * boolean value in SharedPreferences, and may have dependent preferences that are
     33  * enabled/disabled based on the current state.
     34  */
     35 public abstract class TwoStatePreference extends Preference {
     36 
     37     private CharSequence mSummaryOn;
     38     private CharSequence mSummaryOff;
     39     boolean mChecked;
     40     private boolean mSendClickAccessibilityEvent;
     41     private boolean mDisableDependentsState;
     42 
     43 
     44     public TwoStatePreference(Context context, AttributeSet attrs, int defStyle) {
     45         super(context, attrs, defStyle);
     46     }
     47 
     48     public TwoStatePreference(Context context, AttributeSet attrs) {
     49         this(context, attrs, 0);
     50     }
     51 
     52     public TwoStatePreference(Context context) {
     53         this(context, null);
     54     }
     55 
     56     @Override
     57     protected void onClick() {
     58         super.onClick();
     59 
     60         boolean newValue = !isChecked();
     61 
     62         mSendClickAccessibilityEvent = true;
     63 
     64         if (!callChangeListener(newValue)) {
     65             return;
     66         }
     67 
     68         setChecked(newValue);
     69     }
     70 
     71     /**
     72      * Sets the checked state and saves it to the {@link SharedPreferences}.
     73      *
     74      * @param checked The checked state.
     75      */
     76     public void setChecked(boolean checked) {
     77         if (mChecked != checked) {
     78             mChecked = checked;
     79             persistBoolean(checked);
     80             notifyDependencyChange(shouldDisableDependents());
     81             notifyChanged();
     82         }
     83     }
     84 
     85     /**
     86      * Returns the checked state.
     87      *
     88      * @return The checked state.
     89      */
     90     public boolean isChecked() {
     91         return mChecked;
     92     }
     93 
     94     @Override
     95     public boolean shouldDisableDependents() {
     96         boolean shouldDisable = mDisableDependentsState ? mChecked : !mChecked;
     97         return shouldDisable || super.shouldDisableDependents();
     98     }
     99 
    100     /**
    101      * Sets the summary to be shown when checked.
    102      *
    103      * @param summary The summary to be shown when checked.
    104      */
    105     public void setSummaryOn(CharSequence summary) {
    106         mSummaryOn = summary;
    107         if (isChecked()) {
    108             notifyChanged();
    109         }
    110     }
    111 
    112     /**
    113      * @see #setSummaryOn(CharSequence)
    114      * @param summaryResId The summary as a resource.
    115      */
    116     public void setSummaryOn(int summaryResId) {
    117         setSummaryOn(getContext().getString(summaryResId));
    118     }
    119 
    120     /**
    121      * Returns the summary to be shown when checked.
    122      * @return The summary.
    123      */
    124     public CharSequence getSummaryOn() {
    125         return mSummaryOn;
    126     }
    127 
    128     /**
    129      * Sets the summary to be shown when unchecked.
    130      *
    131      * @param summary The summary to be shown when unchecked.
    132      */
    133     public void setSummaryOff(CharSequence summary) {
    134         mSummaryOff = summary;
    135         if (!isChecked()) {
    136             notifyChanged();
    137         }
    138     }
    139 
    140     /**
    141      * @see #setSummaryOff(CharSequence)
    142      * @param summaryResId The summary as a resource.
    143      */
    144     public void setSummaryOff(int summaryResId) {
    145         setSummaryOff(getContext().getString(summaryResId));
    146     }
    147 
    148     /**
    149      * Returns the summary to be shown when unchecked.
    150      * @return The summary.
    151      */
    152     public CharSequence getSummaryOff() {
    153         return mSummaryOff;
    154     }
    155 
    156     /**
    157      * Returns whether dependents are disabled when this preference is on ({@code true})
    158      * or when this preference is off ({@code false}).
    159      *
    160      * @return Whether dependents are disabled when this preference is on ({@code true})
    161      *         or when this preference is off ({@code false}).
    162      */
    163     public boolean getDisableDependentsState() {
    164         return mDisableDependentsState;
    165     }
    166 
    167     /**
    168      * Sets whether dependents are disabled when this preference is on ({@code true})
    169      * or when this preference is off ({@code false}).
    170      *
    171      * @param disableDependentsState The preference state that should disable dependents.
    172      */
    173     public void setDisableDependentsState(boolean disableDependentsState) {
    174         mDisableDependentsState = disableDependentsState;
    175     }
    176 
    177     @Override
    178     protected Object onGetDefaultValue(TypedArray a, int index) {
    179         return a.getBoolean(index, false);
    180     }
    181 
    182     @Override
    183     protected void onSetInitialValue(boolean restoreValue, Object defaultValue) {
    184         setChecked(restoreValue ? getPersistedBoolean(mChecked)
    185                 : (Boolean) defaultValue);
    186     }
    187 
    188     void sendAccessibilityEvent(View view) {
    189         // Since the view is still not attached we create, populate,
    190         // and send the event directly since we do not know when it
    191         // will be attached and posting commands is not as clean.
    192         AccessibilityManager accessibilityManager = AccessibilityManager.getInstance(getContext());
    193         if (mSendClickAccessibilityEvent && accessibilityManager.isEnabled()) {
    194             AccessibilityEvent event = AccessibilityEvent.obtain();
    195             event.setEventType(AccessibilityEvent.TYPE_VIEW_CLICKED);
    196             view.onInitializeAccessibilityEvent(event);
    197             view.dispatchPopulateAccessibilityEvent(event);
    198             accessibilityManager.sendAccessibilityEvent(event);
    199         }
    200         mSendClickAccessibilityEvent = false;
    201     }
    202 
    203     /**
    204      * Sync a summary view contained within view's subhierarchy with the correct summary text.
    205      * @param view View where a summary should be located
    206      */
    207     void syncSummaryView(View view) {
    208         // Sync the summary view
    209         TextView summaryView = (TextView) view.findViewById(com.android.internal.R.id.summary);
    210         if (summaryView != null) {
    211             boolean useDefaultSummary = true;
    212             if (mChecked && mSummaryOn != null) {
    213                 summaryView.setText(mSummaryOn);
    214                 useDefaultSummary = false;
    215             } else if (!mChecked && mSummaryOff != null) {
    216                 summaryView.setText(mSummaryOff);
    217                 useDefaultSummary = false;
    218             }
    219 
    220             if (useDefaultSummary) {
    221                 final CharSequence summary = getSummary();
    222                 if (summary != null) {
    223                     summaryView.setText(summary);
    224                     useDefaultSummary = false;
    225                 }
    226             }
    227 
    228             int newVisibility = View.GONE;
    229             if (!useDefaultSummary) {
    230                 // Someone has written to it
    231                 newVisibility = View.VISIBLE;
    232             }
    233             if (newVisibility != summaryView.getVisibility()) {
    234                 summaryView.setVisibility(newVisibility);
    235             }
    236         }
    237     }
    238 
    239     @Override
    240     protected Parcelable onSaveInstanceState() {
    241         final Parcelable superState = super.onSaveInstanceState();
    242         if (isPersistent()) {
    243             // No need to save instance state since it's persistent
    244             return superState;
    245         }
    246 
    247         final SavedState myState = new SavedState(superState);
    248         myState.checked = isChecked();
    249         return myState;
    250     }
    251 
    252     @Override
    253     protected void onRestoreInstanceState(Parcelable state) {
    254         if (state == null || !state.getClass().equals(SavedState.class)) {
    255             // Didn't save state for us in onSaveInstanceState
    256             super.onRestoreInstanceState(state);
    257             return;
    258         }
    259 
    260         SavedState myState = (SavedState) state;
    261         super.onRestoreInstanceState(myState.getSuperState());
    262         setChecked(myState.checked);
    263     }
    264 
    265     static class SavedState extends BaseSavedState {
    266         boolean checked;
    267 
    268         public SavedState(Parcel source) {
    269             super(source);
    270             checked = source.readInt() == 1;
    271         }
    272 
    273         @Override
    274         public void writeToParcel(Parcel dest, int flags) {
    275             super.writeToParcel(dest, flags);
    276             dest.writeInt(checked ? 1 : 0);
    277         }
    278 
    279         public SavedState(Parcelable superState) {
    280             super(superState);
    281         }
    282 
    283         public static final Parcelable.Creator<SavedState> CREATOR =
    284                 new Parcelable.Creator<SavedState>() {
    285             public SavedState createFromParcel(Parcel in) {
    286                 return new SavedState(in);
    287             }
    288 
    289             public SavedState[] newArray(int size) {
    290                 return new SavedState[size];
    291             }
    292         };
    293     }
    294 }
    295