1 /*
2  * Copyright (C) 2006 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.widget;
18 
19 import android.annotation.ArrayRes;
20 import android.annotation.IdRes;
21 import android.annotation.LayoutRes;
22 import android.annotation.NonNull;
23 import android.annotation.Nullable;
24 import android.compat.annotation.UnsupportedAppUsage;
25 import android.content.Context;
26 import android.content.res.Resources;
27 import android.util.Log;
28 import android.view.ContextThemeWrapper;
29 import android.view.LayoutInflater;
30 import android.view.View;
31 import android.view.ViewGroup;
32 
33 import java.util.ArrayList;
34 import java.util.Arrays;
35 import java.util.Collection;
36 import java.util.Collections;
37 import java.util.Comparator;
38 import java.util.List;
39 
40 /**
41  * You can use this adapter to provide views for an {@link AdapterView},
42  * Returns a view for each object in a collection of data objects you
43  * provide, and can be used with list-based user interface widgets such as
44  * {@link ListView} or {@link Spinner}.
45  * <p>
46  * By default, the array adapter creates a view by calling {@link Object#toString()} on each
47  * data object in the collection you provide, and places the result in a TextView.
48  * You may also customize what type of view is used for the data object in the collection.
49  * To customize what type of view is used for the data object,
50  * override {@link #getView(int, View, ViewGroup)}
51  * and inflate a view resource.
52  * </p>
53  * <p>
54  * For an example of using an array adapter with a ListView, see the
55  * <a href="{@docRoot}guide/topics/ui/declaring-layout.html#AdapterViews">
56  * Adapter Views</a> guide.
57  * </p>
58  * <p>
59  * For an example of using an array adapter with a Spinner, see the
60  * <a href="{@docRoot}guide/topics/ui/controls/spinner.html">Spinners</a> guide.
61  * </p>
62  * <p class="note"><strong>Note:</strong>
63  * If you are considering using array adapter with a ListView, consider using
64  * {@link androidx.recyclerview.widget.RecyclerView} instead.
65  * RecyclerView offers similar features with better performance and more flexibility than
66  * ListView provides.
67  * See the
68  * <a href="https://developer.android.com/guide/topics/ui/layout/recyclerview.html">
69  * Recycler View</a> guide.</p>
70  */
71 public class ArrayAdapter<T> extends BaseAdapter implements Filterable, ThemedSpinnerAdapter {
72     /**
73      * Lock used to modify the content of {@link #mObjects}. Any write operation
74      * performed on the array should be synchronized on this lock. This lock is also
75      * used by the filter (see {@link #getFilter()} to make a synchronized copy of
76      * the original array of data.
77      */
78     @UnsupportedAppUsage
79     private final Object mLock = new Object();
80 
81     private final LayoutInflater mInflater;
82 
83     private final Context mContext;
84 
85     /**
86      * The resource indicating what views to inflate to display the content of this
87      * array adapter.
88      */
89     private final int mResource;
90 
91     /**
92      * The resource indicating what views to inflate to display the content of this
93      * array adapter in a drop down widget.
94      */
95     private int mDropDownResource;
96 
97     /**
98      * Contains the list of objects that represent the data of this ArrayAdapter.
99      * The content of this list is referred to as "the array" in the documentation.
100      */
101     @UnsupportedAppUsage
102     private List<T> mObjects;
103 
104     /**
105      * Indicates whether the contents of {@link #mObjects} came from static resources.
106      */
107     private boolean mObjectsFromResources;
108 
109     /**
110      * If the inflated resource is not a TextView, {@code mFieldId} is used to find
111      * a TextView inside the inflated views hierarchy. This field must contain the
112      * identifier that matches the one defined in the resource file.
113      */
114     private int mFieldId = 0;
115 
116     /**
117      * Indicates whether or not {@link #notifyDataSetChanged()} must be called whenever
118      * {@link #mObjects} is modified.
119      */
120     private boolean mNotifyOnChange = true;
121 
122     // A copy of the original mObjects array, initialized from and then used instead as soon as
123     // the mFilter ArrayFilter is used. mObjects will then only contain the filtered values.
124     @UnsupportedAppUsage
125     private ArrayList<T> mOriginalValues;
126     private ArrayFilter mFilter;
127 
128     /** Layout inflater used for {@link #getDropDownView(int, View, ViewGroup)}. */
129     private LayoutInflater mDropDownInflater;
130 
131     /**
132      * Constructor
133      *
134      * @param context The current context.
135      * @param resource The resource ID for a layout file containing a TextView to use when
136      *                 instantiating views.
137      */
ArrayAdapter(@onNull Context context, @LayoutRes int resource)138     public ArrayAdapter(@NonNull Context context, @LayoutRes int resource) {
139         this(context, resource, 0, new ArrayList<>());
140     }
141 
142     /**
143      * Constructor
144      *
145      * @param context The current context.
146      * @param resource The resource ID for a layout file containing a layout to use when
147      *                 instantiating views.
148      * @param textViewResourceId The id of the TextView within the layout resource to be populated
149      */
ArrayAdapter(@onNull Context context, @LayoutRes int resource, @IdRes int textViewResourceId)150     public ArrayAdapter(@NonNull Context context, @LayoutRes int resource,
151             @IdRes int textViewResourceId) {
152         this(context, resource, textViewResourceId, new ArrayList<>());
153     }
154 
155     /**
156      * Constructor. This constructor will result in the underlying data collection being
157      * immutable, so methods such as {@link #clear()} will throw an exception.
158      *
159      * @param context The current context.
160      * @param resource The resource ID for a layout file containing a TextView to use when
161      *                 instantiating views.
162      * @param objects The objects to represent in the ListView.
163      */
ArrayAdapter(@onNull Context context, @LayoutRes int resource, @NonNull T[] objects)164     public ArrayAdapter(@NonNull Context context, @LayoutRes int resource, @NonNull T[] objects) {
165         this(context, resource, 0, Arrays.asList(objects));
166     }
167 
168     /**
169      * Constructor. This constructor will result in the underlying data collection being
170      * immutable, so methods such as {@link #clear()} will throw an exception.
171      *
172      * @param context The current context.
173      * @param resource The resource ID for a layout file containing a layout to use when
174      *                 instantiating views.
175      * @param textViewResourceId The id of the TextView within the layout resource to be populated
176      * @param objects The objects to represent in the ListView.
177      */
ArrayAdapter(@onNull Context context, @LayoutRes int resource, @IdRes int textViewResourceId, @NonNull T[] objects)178     public ArrayAdapter(@NonNull Context context, @LayoutRes int resource,
179             @IdRes int textViewResourceId, @NonNull T[] objects) {
180         this(context, resource, textViewResourceId, Arrays.asList(objects));
181     }
182 
183     /**
184      * Constructor
185      *
186      * @param context The current context.
187      * @param resource The resource ID for a layout file containing a TextView to use when
188      *                 instantiating views.
189      * @param objects The objects to represent in the ListView.
190      */
ArrayAdapter(@onNull Context context, @LayoutRes int resource, @NonNull List<T> objects)191     public ArrayAdapter(@NonNull Context context, @LayoutRes int resource,
192             @NonNull List<T> objects) {
193         this(context, resource, 0, objects);
194     }
195 
196     /**
197      * Constructor
198      *
199      * @param context The current context.
200      * @param resource The resource ID for a layout file containing a layout to use when
201      *                 instantiating views.
202      * @param textViewResourceId The id of the TextView within the layout resource to be populated
203      * @param objects The objects to represent in the ListView.
204      */
ArrayAdapter(@onNull Context context, @LayoutRes int resource, @IdRes int textViewResourceId, @NonNull List<T> objects)205     public ArrayAdapter(@NonNull Context context, @LayoutRes int resource,
206             @IdRes int textViewResourceId, @NonNull List<T> objects) {
207         this(context, resource, textViewResourceId, objects, false);
208     }
209 
ArrayAdapter(@onNull Context context, @LayoutRes int resource, @IdRes int textViewResourceId, @NonNull List<T> objects, boolean objsFromResources)210     private ArrayAdapter(@NonNull Context context, @LayoutRes int resource,
211             @IdRes int textViewResourceId, @NonNull List<T> objects, boolean objsFromResources) {
212         mContext = context;
213         mInflater = LayoutInflater.from(context);
214         mResource = mDropDownResource = resource;
215         mObjects = objects;
216         mObjectsFromResources = objsFromResources;
217         mFieldId = textViewResourceId;
218     }
219 
220     /**
221      * Adds the specified object at the end of the array.
222      *
223      * @param object The object to add at the end of the array.
224      * @throws UnsupportedOperationException if the underlying data collection is immutable
225      */
add(@ullable T object)226     public void add(@Nullable T object) {
227         synchronized (mLock) {
228             if (mOriginalValues != null) {
229                 mOriginalValues.add(object);
230             } else {
231                 mObjects.add(object);
232             }
233             mObjectsFromResources = false;
234         }
235         if (mNotifyOnChange) notifyDataSetChanged();
236     }
237 
238     /**
239      * Adds the specified Collection at the end of the array.
240      *
241      * @param collection The Collection to add at the end of the array.
242      * @throws UnsupportedOperationException if the <tt>addAll</tt> operation
243      *         is not supported by this list
244      * @throws ClassCastException if the class of an element of the specified
245      *         collection prevents it from being added to this list
246      * @throws NullPointerException if the specified collection contains one
247      *         or more null elements and this list does not permit null
248      *         elements, or if the specified collection is null
249      * @throws IllegalArgumentException if some property of an element of the
250      *         specified collection prevents it from being added to this list
251      */
addAll(@onNull Collection<? extends T> collection)252     public void addAll(@NonNull Collection<? extends T> collection) {
253         synchronized (mLock) {
254             if (mOriginalValues != null) {
255                 mOriginalValues.addAll(collection);
256             } else {
257                 mObjects.addAll(collection);
258             }
259             mObjectsFromResources = false;
260         }
261         if (mNotifyOnChange) notifyDataSetChanged();
262     }
263 
264     /**
265      * Adds the specified items at the end of the array.
266      *
267      * @param items The items to add at the end of the array.
268      * @throws UnsupportedOperationException if the underlying data collection is immutable
269      */
addAll(T .... items)270     public void addAll(T ... items) {
271         synchronized (mLock) {
272             if (mOriginalValues != null) {
273                 Collections.addAll(mOriginalValues, items);
274             } else {
275                 Collections.addAll(mObjects, items);
276             }
277             mObjectsFromResources = false;
278         }
279         if (mNotifyOnChange) notifyDataSetChanged();
280     }
281 
282     /**
283      * Inserts the specified object at the specified index in the array.
284      *
285      * @param object The object to insert into the array.
286      * @param index The index at which the object must be inserted.
287      * @throws UnsupportedOperationException if the underlying data collection is immutable
288      */
insert(@ullable T object, int index)289     public void insert(@Nullable T object, int index) {
290         synchronized (mLock) {
291             if (mOriginalValues != null) {
292                 mOriginalValues.add(index, object);
293             } else {
294                 mObjects.add(index, object);
295             }
296             mObjectsFromResources = false;
297         }
298         if (mNotifyOnChange) notifyDataSetChanged();
299     }
300 
301     /**
302      * Removes the specified object from the array.
303      *
304      * @param object The object to remove.
305      * @throws UnsupportedOperationException if the underlying data collection is immutable
306      */
remove(@ullable T object)307     public void remove(@Nullable T object) {
308         synchronized (mLock) {
309             if (mOriginalValues != null) {
310                 mOriginalValues.remove(object);
311             } else {
312                 mObjects.remove(object);
313             }
314             mObjectsFromResources = false;
315         }
316         if (mNotifyOnChange) notifyDataSetChanged();
317     }
318 
319     /**
320      * Remove all elements from the list.
321      *
322      * @throws UnsupportedOperationException if the underlying data collection is immutable
323      */
clear()324     public void clear() {
325         synchronized (mLock) {
326             if (mOriginalValues != null) {
327                 mOriginalValues.clear();
328             } else {
329                 mObjects.clear();
330             }
331             mObjectsFromResources = false;
332         }
333         if (mNotifyOnChange) notifyDataSetChanged();
334     }
335 
336     /**
337      * Sorts the content of this adapter using the specified comparator.
338      *
339      * @param comparator The comparator used to sort the objects contained
340      *        in this adapter.
341      */
sort(@onNull Comparator<? super T> comparator)342     public void sort(@NonNull Comparator<? super T> comparator) {
343         synchronized (mLock) {
344             if (mOriginalValues != null) {
345                 Collections.sort(mOriginalValues, comparator);
346             } else {
347                 Collections.sort(mObjects, comparator);
348             }
349         }
350         if (mNotifyOnChange) notifyDataSetChanged();
351     }
352 
353     @Override
notifyDataSetChanged()354     public void notifyDataSetChanged() {
355         super.notifyDataSetChanged();
356         mNotifyOnChange = true;
357     }
358 
359     /**
360      * Control whether methods that change the list ({@link #add}, {@link #addAll(Collection)},
361      * {@link #addAll(Object[])}, {@link #insert}, {@link #remove}, {@link #clear},
362      * {@link #sort(Comparator)}) automatically call {@link #notifyDataSetChanged}.  If set to
363      * false, caller must manually call notifyDataSetChanged() to have the changes
364      * reflected in the attached view.
365      *
366      * The default is true, and calling notifyDataSetChanged()
367      * resets the flag to true.
368      *
369      * @param notifyOnChange if true, modifications to the list will
370      *                       automatically call {@link
371      *                       #notifyDataSetChanged}
372      */
setNotifyOnChange(boolean notifyOnChange)373     public void setNotifyOnChange(boolean notifyOnChange) {
374         mNotifyOnChange = notifyOnChange;
375     }
376 
377     /**
378      * Returns the context associated with this array adapter. The context is used
379      * to create views from the resource passed to the constructor.
380      *
381      * @return The Context associated with this adapter.
382      */
getContext()383     public @NonNull Context getContext() {
384         return mContext;
385     }
386 
387     @Override
getCount()388     public int getCount() {
389         return mObjects.size();
390     }
391 
392     @Override
getItem(int position)393     public @Nullable T getItem(int position) {
394         return mObjects.get(position);
395     }
396 
397     /**
398      * Returns the position of the specified item in the array.
399      *
400      * @param item The item to retrieve the position of.
401      *
402      * @return The position of the specified item.
403      */
getPosition(@ullable T item)404     public int getPosition(@Nullable T item) {
405         return mObjects.indexOf(item);
406     }
407 
408     @Override
getItemId(int position)409     public long getItemId(int position) {
410         return position;
411     }
412 
413     @Override
getView(int position, @Nullable View convertView, @NonNull ViewGroup parent)414     public @NonNull View getView(int position, @Nullable View convertView,
415             @NonNull ViewGroup parent) {
416         return createViewFromResource(mInflater, position, convertView, parent, mResource);
417     }
418 
createViewFromResource(@onNull LayoutInflater inflater, int position, @Nullable View convertView, @NonNull ViewGroup parent, int resource)419     private @NonNull View createViewFromResource(@NonNull LayoutInflater inflater, int position,
420             @Nullable View convertView, @NonNull ViewGroup parent, int resource) {
421         final View view;
422         final TextView text;
423 
424         if (convertView == null) {
425             view = inflater.inflate(resource, parent, false);
426         } else {
427             view = convertView;
428         }
429 
430         try {
431             if (mFieldId == 0) {
432                 //  If no custom field is assigned, assume the whole resource is a TextView
433                 text = (TextView) view;
434             } else {
435                 //  Otherwise, find the TextView field within the layout
436                 text = view.findViewById(mFieldId);
437 
438                 if (text == null) {
439                     throw new RuntimeException("Failed to find view with ID "
440                             + mContext.getResources().getResourceName(mFieldId)
441                             + " in item layout");
442                 }
443             }
444         } catch (ClassCastException e) {
445             Log.e("ArrayAdapter", "You must supply a resource ID for a TextView");
446             throw new IllegalStateException(
447                     "ArrayAdapter requires the resource ID to be a TextView", e);
448         }
449 
450         final T item = getItem(position);
451         if (item instanceof CharSequence) {
452             text.setText((CharSequence) item);
453         } else {
454             text.setText(item.toString());
455         }
456 
457         return view;
458     }
459 
460     /**
461      * <p>Sets the layout resource to create the drop down views.</p>
462      *
463      * @param resource the layout resource defining the drop down views
464      * @see #getDropDownView(int, android.view.View, android.view.ViewGroup)
465      */
setDropDownViewResource(@ayoutRes int resource)466     public void setDropDownViewResource(@LayoutRes int resource) {
467         this.mDropDownResource = resource;
468     }
469 
470     /**
471      * Sets the {@link Resources.Theme} against which drop-down views are
472      * inflated.
473      * <p>
474      * By default, drop-down views are inflated against the theme of the
475      * {@link Context} passed to the adapter's constructor.
476      *
477      * @param theme the theme against which to inflate drop-down views or
478      *              {@code null} to use the theme from the adapter's context
479      * @see #getDropDownView(int, View, ViewGroup)
480      */
481     @Override
setDropDownViewTheme(@ullable Resources.Theme theme)482     public void setDropDownViewTheme(@Nullable Resources.Theme theme) {
483         if (theme == null) {
484             mDropDownInflater = null;
485         } else if (theme == mInflater.getContext().getTheme()) {
486             mDropDownInflater = mInflater;
487         } else {
488             final Context context = new ContextThemeWrapper(mContext, theme);
489             mDropDownInflater = LayoutInflater.from(context);
490         }
491     }
492 
493     @Override
getDropDownViewTheme()494     public @Nullable Resources.Theme getDropDownViewTheme() {
495         return mDropDownInflater == null ? null : mDropDownInflater.getContext().getTheme();
496     }
497 
498     @Override
getDropDownView(int position, @Nullable View convertView, @NonNull ViewGroup parent)499     public View getDropDownView(int position, @Nullable View convertView,
500             @NonNull ViewGroup parent) {
501         final LayoutInflater inflater = mDropDownInflater == null ? mInflater : mDropDownInflater;
502         return createViewFromResource(inflater, position, convertView, parent, mDropDownResource);
503     }
504 
505     /**
506      * Creates a new ArrayAdapter from external resources. The content of the array is
507      * obtained through {@link android.content.res.Resources#getTextArray(int)}.
508      *
509      * @param context The application's environment.
510      * @param textArrayResId The identifier of the array to use as the data source.
511      * @param textViewResId The identifier of the layout used to create views.
512      *
513      * @return An ArrayAdapter<CharSequence>.
514      */
createFromResource(@onNull Context context, @ArrayRes int textArrayResId, @LayoutRes int textViewResId)515     public static @NonNull ArrayAdapter<CharSequence> createFromResource(@NonNull Context context,
516             @ArrayRes int textArrayResId, @LayoutRes int textViewResId) {
517         final CharSequence[] strings = context.getResources().getTextArray(textArrayResId);
518         return new ArrayAdapter<>(context, textViewResId, 0, Arrays.asList(strings), true);
519     }
520 
521     @Override
getFilter()522     public @NonNull Filter getFilter() {
523         if (mFilter == null) {
524             mFilter = new ArrayFilter();
525         }
526         return mFilter;
527     }
528 
529     /**
530      * {@inheritDoc}
531      *
532      * @return values from the string array used by {@link #createFromResource(Context, int, int)},
533      * or {@code null} if object was created otherwsie or if contents were dynamically changed after
534      * creation.
535      */
536     @Override
getAutofillOptions()537     public CharSequence[] getAutofillOptions() {
538         // First check if app developer explicitly set them.
539         final CharSequence[] explicitOptions = super.getAutofillOptions();
540         if (explicitOptions != null) {
541             return explicitOptions;
542         }
543 
544         // Otherwise, only return options that came from static resources.
545         if (!mObjectsFromResources || mObjects == null || mObjects.isEmpty()) {
546             return null;
547         }
548         final int size = mObjects.size();
549         final CharSequence[] options = new CharSequence[size];
550         mObjects.toArray(options);
551         return options;
552     }
553 
554     /**
555      * <p>An array filter constrains the content of the array adapter with
556      * a prefix. Each item that does not start with the supplied prefix
557      * is removed from the list.</p>
558      */
559     private class ArrayFilter extends Filter {
560         @Override
performFiltering(CharSequence prefix)561         protected FilterResults performFiltering(CharSequence prefix) {
562             final FilterResults results = new FilterResults();
563 
564             if (mOriginalValues == null) {
565                 synchronized (mLock) {
566                     mOriginalValues = new ArrayList<>(mObjects);
567                 }
568             }
569 
570             if (prefix == null || prefix.length() == 0) {
571                 final ArrayList<T> list;
572                 synchronized (mLock) {
573                     list = new ArrayList<>(mOriginalValues);
574                 }
575                 results.values = list;
576                 results.count = list.size();
577             } else {
578                 final String prefixString = prefix.toString().toLowerCase();
579 
580                 final ArrayList<T> values;
581                 synchronized (mLock) {
582                     values = new ArrayList<>(mOriginalValues);
583                 }
584 
585                 final int count = values.size();
586                 final ArrayList<T> newValues = new ArrayList<>();
587 
588                 for (int i = 0; i < count; i++) {
589                     final T value = values.get(i);
590                     final String valueText = value.toString().toLowerCase();
591 
592                     // First match against the whole, non-splitted value
593                     if (valueText.startsWith(prefixString)) {
594                         newValues.add(value);
595                     } else {
596                         final String[] words = valueText.split(" ");
597                         for (String word : words) {
598                             if (word.startsWith(prefixString)) {
599                                 newValues.add(value);
600                                 break;
601                             }
602                         }
603                     }
604                 }
605 
606                 results.values = newValues;
607                 results.count = newValues.size();
608             }
609 
610             return results;
611         }
612 
613         @Override
publishResults(CharSequence constraint, FilterResults results)614         protected void publishResults(CharSequence constraint, FilterResults results) {
615             //noinspection unchecked
616             mObjects = (List<T>) results.values;
617             if (results.count > 0) {
618                 notifyDataSetChanged();
619             } else {
620                 notifyDataSetInvalidated();
621             }
622         }
623     }
624 }
625