1 /*
2  * Copyright (C) 2007 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 java.util.ArrayList;
20 import java.util.Collections;
21 import java.util.List;
22 
23 import android.graphics.drawable.Drawable;
24 import android.os.Handler;
25 import android.preference.Preference.OnPreferenceChangeInternalListener;
26 import android.view.View;
27 import android.view.ViewGroup;
28 import android.widget.Adapter;
29 import android.widget.BaseAdapter;
30 import android.widget.FrameLayout;
31 import android.widget.ListView;
32 
33 /**
34  * An adapter that returns the {@link Preference} contained in this group.
35  * In most cases, this adapter should be the base class for any custom
36  * adapters from {@link Preference#getAdapter()}.
37  * <p>
38  * This adapter obeys the
39  * {@link Preference}'s adapter rule (the
40  * {@link Adapter#getView(int, View, ViewGroup)} should be used instead of
41  * {@link Preference#getView(ViewGroup)} if a {@link Preference} has an
42  * adapter via {@link Preference#getAdapter()}).
43  * <p>
44  * This adapter also propagates data change/invalidated notifications upward.
45  * <p>
46  * This adapter does not include this {@link PreferenceGroup} in the returned
47  * adapter, use {@link PreferenceCategoryAdapter} instead.
48  *
49  * @see PreferenceCategoryAdapter
50  *
51  * @hide
52  */
53 public class PreferenceGroupAdapter extends BaseAdapter
54         implements OnPreferenceChangeInternalListener {
55 
56     private static final String TAG = "PreferenceGroupAdapter";
57 
58     /**
59      * The group that we are providing data from.
60      */
61     private PreferenceGroup mPreferenceGroup;
62 
63     /**
64      * Maps a position into this adapter -> {@link Preference}. These
65      * {@link Preference}s don't have to be direct children of this
66      * {@link PreferenceGroup}, they can be grand children or younger)
67      */
68     private List<Preference> mPreferenceList;
69 
70     /**
71      * List of unique Preference and its subclasses' names. This is used to find
72      * out how many types of views this adapter can return. Once the count is
73      * returned, this cannot be modified (since the ListView only checks the
74      * count once--when the adapter is being set). We will not recycle views for
75      * Preference subclasses seen after the count has been returned.
76      */
77     private ArrayList<PreferenceLayout> mPreferenceLayouts;
78 
79     private PreferenceLayout mTempPreferenceLayout = new PreferenceLayout();
80 
81     /**
82      * Blocks the mPreferenceClassNames from being changed anymore.
83      */
84     private boolean mHasReturnedViewTypeCount = false;
85 
86     private volatile boolean mIsSyncing = false;
87 
88     private Handler mHandler = new Handler();
89 
90     private Runnable mSyncRunnable = new Runnable() {
91         public void run() {
92             syncMyPreferences();
93         }
94     };
95 
96     private int mHighlightedPosition = -1;
97     private Drawable mHighlightedDrawable;
98 
99     private static ViewGroup.LayoutParams sWrapperLayoutParams = new ViewGroup.LayoutParams(
100             ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
101 
102     private static class PreferenceLayout implements Comparable<PreferenceLayout> {
103         private int resId;
104         private int widgetResId;
105         private String name;
106 
compareTo(PreferenceLayout other)107         public int compareTo(PreferenceLayout other) {
108             int compareNames = name.compareTo(other.name);
109             if (compareNames == 0) {
110                 if (resId == other.resId) {
111                     if (widgetResId == other.widgetResId) {
112                         return 0;
113                     } else {
114                         return widgetResId - other.widgetResId;
115                     }
116                 } else {
117                     return resId - other.resId;
118                 }
119             } else {
120                 return compareNames;
121             }
122         }
123     }
124 
PreferenceGroupAdapter(PreferenceGroup preferenceGroup)125     public PreferenceGroupAdapter(PreferenceGroup preferenceGroup) {
126         mPreferenceGroup = preferenceGroup;
127         // If this group gets or loses any children, let us know
128         mPreferenceGroup.setOnPreferenceChangeInternalListener(this);
129 
130         mPreferenceList = new ArrayList<Preference>();
131         mPreferenceLayouts = new ArrayList<PreferenceLayout>();
132 
133         syncMyPreferences();
134     }
135 
syncMyPreferences()136     private void syncMyPreferences() {
137         synchronized(this) {
138             if (mIsSyncing) {
139                 return;
140             }
141 
142             mIsSyncing = true;
143         }
144 
145         List<Preference> newPreferenceList = new ArrayList<Preference>(mPreferenceList.size());
146         flattenPreferenceGroup(newPreferenceList, mPreferenceGroup);
147         mPreferenceList = newPreferenceList;
148 
149         notifyDataSetChanged();
150 
151         synchronized(this) {
152             mIsSyncing = false;
153             notifyAll();
154         }
155     }
156 
flattenPreferenceGroup(List<Preference> preferences, PreferenceGroup group)157     private void flattenPreferenceGroup(List<Preference> preferences, PreferenceGroup group) {
158         // TODO: shouldn't always?
159         group.sortPreferences();
160 
161         final int groupSize = group.getPreferenceCount();
162         for (int i = 0; i < groupSize; i++) {
163             final Preference preference = group.getPreference(i);
164 
165             preferences.add(preference);
166 
167             if (!mHasReturnedViewTypeCount && preference.isRecycleEnabled()) {
168                 addPreferenceClassName(preference);
169             }
170 
171             if (preference instanceof PreferenceGroup) {
172                 final PreferenceGroup preferenceAsGroup = (PreferenceGroup) preference;
173                 if (preferenceAsGroup.isOnSameScreenAsChildren()) {
174                     flattenPreferenceGroup(preferences, preferenceAsGroup);
175                 }
176             }
177 
178             preference.setOnPreferenceChangeInternalListener(this);
179         }
180     }
181 
182     /**
183      * Creates a string that includes the preference name, layout id and widget layout id.
184      * If a particular preference type uses 2 different resources, they will be treated as
185      * different view types.
186      */
createPreferenceLayout(Preference preference, PreferenceLayout in)187     private PreferenceLayout createPreferenceLayout(Preference preference, PreferenceLayout in) {
188         PreferenceLayout pl = in != null? in : new PreferenceLayout();
189         pl.name = preference.getClass().getName();
190         pl.resId = preference.getLayoutResource();
191         pl.widgetResId = preference.getWidgetLayoutResource();
192         return pl;
193     }
194 
addPreferenceClassName(Preference preference)195     private void addPreferenceClassName(Preference preference) {
196         final PreferenceLayout pl = createPreferenceLayout(preference, null);
197         int insertPos = Collections.binarySearch(mPreferenceLayouts, pl);
198 
199         // Only insert if it doesn't exist (when it is negative).
200         if (insertPos < 0) {
201             // Convert to insert index
202             insertPos = insertPos * -1 - 1;
203             mPreferenceLayouts.add(insertPos, pl);
204         }
205     }
206 
getCount()207     public int getCount() {
208         return mPreferenceList.size();
209     }
210 
getItem(int position)211     public Preference getItem(int position) {
212         if (position < 0 || position >= getCount()) return null;
213         return mPreferenceList.get(position);
214     }
215 
getItemId(int position)216     public long getItemId(int position) {
217         if (position < 0 || position >= getCount()) return ListView.INVALID_ROW_ID;
218         return this.getItem(position).getId();
219     }
220 
221     /**
222      * @hide
223      */
setHighlighted(int position)224     public void setHighlighted(int position) {
225         mHighlightedPosition = position;
226     }
227 
228     /**
229      * @hide
230      */
setHighlightedDrawable(Drawable drawable)231     public void setHighlightedDrawable(Drawable drawable) {
232         mHighlightedDrawable = drawable;
233     }
234 
getView(int position, View convertView, ViewGroup parent)235     public View getView(int position, View convertView, ViewGroup parent) {
236         final Preference preference = this.getItem(position);
237         // Build a PreferenceLayout to compare with known ones that are cacheable.
238         mTempPreferenceLayout = createPreferenceLayout(preference, mTempPreferenceLayout);
239 
240         // If it's not one of the cached ones, set the convertView to null so that
241         // the layout gets re-created by the Preference.
242         if (Collections.binarySearch(mPreferenceLayouts, mTempPreferenceLayout) < 0 ||
243                 (getItemViewType(position) == getHighlightItemViewType())) {
244             convertView = null;
245         }
246         View result = preference.getView(convertView, parent);
247         if (position == mHighlightedPosition && mHighlightedDrawable != null) {
248             ViewGroup wrapper = new FrameLayout(parent.getContext());
249             wrapper.setLayoutParams(sWrapperLayoutParams);
250             wrapper.setBackgroundDrawable(mHighlightedDrawable);
251             wrapper.addView(result);
252             result = wrapper;
253         }
254         return result;
255     }
256 
257     @Override
isEnabled(int position)258     public boolean isEnabled(int position) {
259         if (position < 0 || position >= getCount()) return true;
260         return this.getItem(position).isSelectable();
261     }
262 
263     @Override
areAllItemsEnabled()264     public boolean areAllItemsEnabled() {
265         // There should always be a preference group, and these groups are always
266         // disabled
267         return false;
268     }
269 
onPreferenceChange(Preference preference)270     public void onPreferenceChange(Preference preference) {
271         notifyDataSetChanged();
272     }
273 
onPreferenceHierarchyChange(Preference preference)274     public void onPreferenceHierarchyChange(Preference preference) {
275         mHandler.removeCallbacks(mSyncRunnable);
276         mHandler.post(mSyncRunnable);
277     }
278 
279     @Override
hasStableIds()280     public boolean hasStableIds() {
281         return true;
282     }
283 
getHighlightItemViewType()284     private int getHighlightItemViewType() {
285         return getViewTypeCount() - 1;
286     }
287 
288     @Override
getItemViewType(int position)289     public int getItemViewType(int position) {
290         if (position == mHighlightedPosition) {
291             return getHighlightItemViewType();
292         }
293 
294         if (!mHasReturnedViewTypeCount) {
295             mHasReturnedViewTypeCount = true;
296         }
297 
298         final Preference preference = this.getItem(position);
299         if (!preference.isRecycleEnabled()) {
300             return IGNORE_ITEM_VIEW_TYPE;
301         }
302 
303         mTempPreferenceLayout = createPreferenceLayout(preference, mTempPreferenceLayout);
304 
305         int viewType = Collections.binarySearch(mPreferenceLayouts, mTempPreferenceLayout);
306         if (viewType < 0) {
307             // This is a class that was seen after we returned the count, so
308             // don't recycle it.
309             return IGNORE_ITEM_VIEW_TYPE;
310         } else {
311             return viewType;
312         }
313     }
314 
315     @Override
getViewTypeCount()316     public int getViewTypeCount() {
317         if (!mHasReturnedViewTypeCount) {
318             mHasReturnedViewTypeCount = true;
319         }
320 
321         return Math.max(1, mPreferenceLayouts.size()) + 1;
322     }
323 
324 }
325