• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2009 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.app.SearchDialog;
20 import android.app.SearchManager;
21 import android.app.SearchableInfo;
22 import android.content.ComponentName;
23 import android.content.ContentResolver;
24 import android.content.Context;
25 import android.content.ContentResolver.OpenResourceIdResult;
26 import android.content.pm.ActivityInfo;
27 import android.content.pm.PackageManager;
28 import android.content.pm.PackageManager.NameNotFoundException;
29 import android.content.res.ColorStateList;
30 import android.content.res.Resources;
31 import android.database.Cursor;
32 import android.graphics.drawable.Drawable;
33 import android.net.Uri;
34 import android.os.Bundle;
35 import android.text.Spannable;
36 import android.text.SpannableString;
37 import android.text.TextUtils;
38 import android.text.style.TextAppearanceSpan;
39 import android.util.Log;
40 import android.util.TypedValue;
41 import android.view.View;
42 import android.view.ViewGroup;
43 import android.view.View.OnClickListener;
44 
45 import com.android.internal.R;
46 
47 import java.io.FileNotFoundException;
48 import java.io.IOException;
49 import java.io.InputStream;
50 import java.util.WeakHashMap;
51 
52 /**
53  * Provides the contents for the suggestion drop-down list.in {@link SearchDialog}.
54  *
55  * @hide
56  */
57 class SuggestionsAdapter extends ResourceCursorAdapter implements OnClickListener {
58 
59     private static final boolean DBG = false;
60     private static final String LOG_TAG = "SuggestionsAdapter";
61     private static final int QUERY_LIMIT = 50;
62 
63     static final int REFINE_NONE = 0;
64     static final int REFINE_BY_ENTRY = 1;
65     static final int REFINE_ALL = 2;
66 
67     private SearchManager mSearchManager;
68     private SearchView mSearchView;
69     private SearchableInfo mSearchable;
70     private Context mProviderContext;
71     private WeakHashMap<String, Drawable.ConstantState> mOutsideDrawablesCache;
72     private boolean mClosed = false;
73     private int mQueryRefinement = REFINE_BY_ENTRY;
74 
75     // URL color
76     private ColorStateList mUrlColor;
77 
78     static final int INVALID_INDEX = -1;
79 
80     // Cached column indexes, updated when the cursor changes.
81     private int mText1Col = INVALID_INDEX;
82     private int mText2Col = INVALID_INDEX;
83     private int mText2UrlCol = INVALID_INDEX;
84     private int mIconName1Col = INVALID_INDEX;
85     private int mIconName2Col = INVALID_INDEX;
86     private int mFlagsCol = INVALID_INDEX;
87 
88     // private final Runnable mStartSpinnerRunnable;
89     // private final Runnable mStopSpinnerRunnable;
90 
91     /**
92      * The amount of time we delay in the filter when the user presses the delete key.
93      * @see Filter#setDelayer(android.widget.Filter.Delayer).
94      */
95     private static final long DELETE_KEY_POST_DELAY = 500L;
96 
SuggestionsAdapter(Context context, SearchView searchView, SearchableInfo searchable, WeakHashMap<String, Drawable.ConstantState> outsideDrawablesCache)97     public SuggestionsAdapter(Context context, SearchView searchView,
98             SearchableInfo searchable,
99             WeakHashMap<String, Drawable.ConstantState> outsideDrawablesCache) {
100         super(context,
101                 com.android.internal.R.layout.search_dropdown_item_icons_2line,
102                 null,   // no initial cursor
103                 true);  // auto-requery
104         mSearchManager = (SearchManager) mContext.getSystemService(Context.SEARCH_SERVICE);
105         mSearchView = searchView;
106         mSearchable = searchable;
107         // set up provider resources (gives us icons, etc.)
108         Context activityContext = mSearchable.getActivityContext(mContext);
109         mProviderContext = mSearchable.getProviderContext(mContext, activityContext);
110 
111         mOutsideDrawablesCache = outsideDrawablesCache;
112 
113         // mStartSpinnerRunnable = new Runnable() {
114         // public void run() {
115         // // mSearchView.setWorking(true); // TODO:
116         // }
117         // };
118         //
119         // mStopSpinnerRunnable = new Runnable() {
120         // public void run() {
121         // // mSearchView.setWorking(false); // TODO:
122         // }
123         // };
124 
125         // delay 500ms when deleting
126         getFilter().setDelayer(new Filter.Delayer() {
127 
128             private int mPreviousLength = 0;
129 
130             public long getPostingDelay(CharSequence constraint) {
131                 if (constraint == null) return 0;
132 
133                 long delay = constraint.length() < mPreviousLength ? DELETE_KEY_POST_DELAY : 0;
134                 mPreviousLength = constraint.length();
135                 return delay;
136             }
137         });
138     }
139 
140     /**
141      * Enables query refinement for all suggestions. This means that an additional icon
142      * will be shown for each entry. When clicked, the suggested text on that line will be
143      * copied to the query text field.
144      * <p>
145      *
146      * @param refine which queries to refine. Possible values are {@link #REFINE_NONE},
147      * {@link #REFINE_BY_ENTRY}, and {@link #REFINE_ALL}.
148      */
setQueryRefinement(int refineWhat)149     public void setQueryRefinement(int refineWhat) {
150         mQueryRefinement = refineWhat;
151     }
152 
153     /**
154      * Returns the current query refinement preference.
155      * @return value of query refinement preference
156      */
getQueryRefinement()157     public int getQueryRefinement() {
158         return mQueryRefinement;
159     }
160 
161     /**
162      * Overridden to always return <code>false</code>, since we cannot be sure that
163      * suggestion sources return stable IDs.
164      */
165     @Override
hasStableIds()166     public boolean hasStableIds() {
167         return false;
168     }
169 
170     /**
171      * Use the search suggestions provider to obtain a live cursor.  This will be called
172      * in a worker thread, so it's OK if the query is slow (e.g. round trip for suggestions).
173      * The results will be processed in the UI thread and changeCursor() will be called.
174      */
175     @Override
runQueryOnBackgroundThread(CharSequence constraint)176     public Cursor runQueryOnBackgroundThread(CharSequence constraint) {
177         if (DBG) Log.d(LOG_TAG, "runQueryOnBackgroundThread(" + constraint + ")");
178         String query = (constraint == null) ? "" : constraint.toString();
179         /**
180          * for in app search we show the progress spinner until the cursor is returned with
181          * the results.
182          */
183         Cursor cursor = null;
184         if (mSearchView.getVisibility() != View.VISIBLE
185                 || mSearchView.getWindowVisibility() != View.VISIBLE) {
186             return null;
187         }
188         //mSearchView.getWindow().getDecorView().post(mStartSpinnerRunnable); // TODO:
189         try {
190             cursor = mSearchManager.getSuggestions(mSearchable, query, QUERY_LIMIT);
191             // trigger fill window so the spinner stays up until the results are copied over and
192             // closer to being ready
193             if (cursor != null) {
194                 cursor.getCount();
195                 return cursor;
196             }
197         } catch (RuntimeException e) {
198             Log.w(LOG_TAG, "Search suggestions query threw an exception.", e);
199         }
200         // If cursor is null or an exception was thrown, stop the spinner and return null.
201         // changeCursor doesn't get called if cursor is null
202         // mSearchView.getWindow().getDecorView().post(mStopSpinnerRunnable); // TODO:
203         return null;
204     }
205 
close()206     public void close() {
207         if (DBG) Log.d(LOG_TAG, "close()");
208         changeCursor(null);
209         mClosed = true;
210     }
211 
212     @Override
notifyDataSetChanged()213     public void notifyDataSetChanged() {
214         if (DBG) Log.d(LOG_TAG, "notifyDataSetChanged");
215         super.notifyDataSetChanged();
216 
217         // mSearchView.onDataSetChanged(); // TODO:
218 
219         updateSpinnerState(getCursor());
220     }
221 
222     @Override
notifyDataSetInvalidated()223     public void notifyDataSetInvalidated() {
224         if (DBG) Log.d(LOG_TAG, "notifyDataSetInvalidated");
225         super.notifyDataSetInvalidated();
226 
227         updateSpinnerState(getCursor());
228     }
229 
updateSpinnerState(Cursor cursor)230     private void updateSpinnerState(Cursor cursor) {
231         Bundle extras = cursor != null ? cursor.getExtras() : null;
232         if (DBG) {
233             Log.d(LOG_TAG, "updateSpinnerState - extra = "
234                 + (extras != null
235                         ? extras.getBoolean(SearchManager.CURSOR_EXTRA_KEY_IN_PROGRESS)
236                         : null));
237         }
238         // Check if the Cursor indicates that the query is not complete and show the spinner
239         if (extras != null
240                 && extras.getBoolean(SearchManager.CURSOR_EXTRA_KEY_IN_PROGRESS)) {
241             // mSearchView.getWindow().getDecorView().post(mStartSpinnerRunnable); // TODO:
242             return;
243         }
244         // If cursor is null or is done, stop the spinner
245         // mSearchView.getWindow().getDecorView().post(mStopSpinnerRunnable); // TODO:
246     }
247 
248     /**
249      * Cache columns.
250      */
251     @Override
changeCursor(Cursor c)252     public void changeCursor(Cursor c) {
253         if (DBG) Log.d(LOG_TAG, "changeCursor(" + c + ")");
254 
255         if (mClosed) {
256             Log.w(LOG_TAG, "Tried to change cursor after adapter was closed.");
257             if (c != null) c.close();
258             return;
259         }
260 
261         try {
262             super.changeCursor(c);
263 
264             if (c != null) {
265                 mText1Col = c.getColumnIndex(SearchManager.SUGGEST_COLUMN_TEXT_1);
266                 mText2Col = c.getColumnIndex(SearchManager.SUGGEST_COLUMN_TEXT_2);
267                 mText2UrlCol = c.getColumnIndex(SearchManager.SUGGEST_COLUMN_TEXT_2_URL);
268                 mIconName1Col = c.getColumnIndex(SearchManager.SUGGEST_COLUMN_ICON_1);
269                 mIconName2Col = c.getColumnIndex(SearchManager.SUGGEST_COLUMN_ICON_2);
270                 mFlagsCol = c.getColumnIndex(SearchManager.SUGGEST_COLUMN_FLAGS);
271             }
272         } catch (Exception e) {
273             Log.e(LOG_TAG, "error changing cursor and caching columns", e);
274         }
275     }
276 
277     /**
278      * Tags the view with cached child view look-ups.
279      */
280     @Override
newView(Context context, Cursor cursor, ViewGroup parent)281     public View newView(Context context, Cursor cursor, ViewGroup parent) {
282         View v = super.newView(context, cursor, parent);
283         v.setTag(new ChildViewCache(v));
284         return v;
285     }
286 
287     /**
288      * Cache of the child views of drop-drown list items, to avoid looking up the children
289      * each time the contents of a list item are changed.
290      */
291     private final static class ChildViewCache {
292         public final TextView mText1;
293         public final TextView mText2;
294         public final ImageView mIcon1;
295         public final ImageView mIcon2;
296         public final ImageView mIconRefine;
297 
ChildViewCache(View v)298         public ChildViewCache(View v) {
299             mText1 = (TextView) v.findViewById(com.android.internal.R.id.text1);
300             mText2 = (TextView) v.findViewById(com.android.internal.R.id.text2);
301             mIcon1 = (ImageView) v.findViewById(com.android.internal.R.id.icon1);
302             mIcon2 = (ImageView) v.findViewById(com.android.internal.R.id.icon2);
303             mIconRefine = (ImageView) v.findViewById(com.android.internal.R.id.edit_query);
304         }
305     }
306 
307     @Override
bindView(View view, Context context, Cursor cursor)308     public void bindView(View view, Context context, Cursor cursor) {
309         ChildViewCache views = (ChildViewCache) view.getTag();
310 
311         int flags = 0;
312         if (mFlagsCol != INVALID_INDEX) {
313             flags = cursor.getInt(mFlagsCol);
314         }
315         if (views.mText1 != null) {
316             String text1 = getStringOrNull(cursor, mText1Col);
317             setViewText(views.mText1, text1);
318         }
319         if (views.mText2 != null) {
320             // First check TEXT_2_URL
321             CharSequence text2 = getStringOrNull(cursor, mText2UrlCol);
322             if (text2 != null) {
323                 text2 = formatUrl(text2);
324             } else {
325                 text2 = getStringOrNull(cursor, mText2Col);
326             }
327 
328             // If no second line of text is indicated, allow the first line of text
329             // to be up to two lines if it wants to be.
330             if (TextUtils.isEmpty(text2)) {
331                 if (views.mText1 != null) {
332                     views.mText1.setSingleLine(false);
333                     views.mText1.setMaxLines(2);
334                 }
335             } else {
336                 if (views.mText1 != null) {
337                     views.mText1.setSingleLine(true);
338                     views.mText1.setMaxLines(1);
339                 }
340             }
341             setViewText(views.mText2, text2);
342         }
343 
344         if (views.mIcon1 != null) {
345             setViewDrawable(views.mIcon1, getIcon1(cursor), View.INVISIBLE);
346         }
347         if (views.mIcon2 != null) {
348             setViewDrawable(views.mIcon2, getIcon2(cursor), View.GONE);
349         }
350         if (mQueryRefinement == REFINE_ALL
351                 || (mQueryRefinement == REFINE_BY_ENTRY
352                         && (flags & SearchManager.FLAG_QUERY_REFINEMENT) != 0)) {
353             views.mIconRefine.setVisibility(View.VISIBLE);
354             views.mIconRefine.setTag(views.mText1.getText());
355             views.mIconRefine.setOnClickListener(this);
356         } else {
357             views.mIconRefine.setVisibility(View.GONE);
358         }
359     }
360 
onClick(View v)361     public void onClick(View v) {
362         Object tag = v.getTag();
363         if (tag instanceof CharSequence) {
364             mSearchView.onQueryRefine((CharSequence) tag);
365         }
366     }
367 
formatUrl(CharSequence url)368     private CharSequence formatUrl(CharSequence url) {
369         if (mUrlColor == null) {
370             // Lazily get the URL color from the current theme.
371             TypedValue colorValue = new TypedValue();
372             mContext.getTheme().resolveAttribute(R.attr.textColorSearchUrl, colorValue, true);
373             mUrlColor = mContext.getResources().getColorStateList(colorValue.resourceId);
374         }
375 
376         SpannableString text = new SpannableString(url);
377         text.setSpan(new TextAppearanceSpan(null, 0, 0, mUrlColor, null),
378                 0, url.length(),
379                 Spannable.SPAN_EXCLUSIVE_EXCLUSIVE);
380         return text;
381     }
382 
setViewText(TextView v, CharSequence text)383     private void setViewText(TextView v, CharSequence text) {
384         // Set the text even if it's null, since we need to clear any previous text.
385         v.setText(text);
386 
387         if (TextUtils.isEmpty(text)) {
388             v.setVisibility(View.GONE);
389         } else {
390             v.setVisibility(View.VISIBLE);
391         }
392     }
393 
getIcon1(Cursor cursor)394     private Drawable getIcon1(Cursor cursor) {
395         if (mIconName1Col == INVALID_INDEX) {
396             return null;
397         }
398         String value = cursor.getString(mIconName1Col);
399         Drawable drawable = getDrawableFromResourceValue(value);
400         if (drawable != null) {
401             return drawable;
402         }
403         return getDefaultIcon1(cursor);
404     }
405 
getIcon2(Cursor cursor)406     private Drawable getIcon2(Cursor cursor) {
407         if (mIconName2Col == INVALID_INDEX) {
408             return null;
409         }
410         String value = cursor.getString(mIconName2Col);
411         return getDrawableFromResourceValue(value);
412     }
413 
414     /**
415      * Sets the drawable in an image view, makes sure the view is only visible if there
416      * is a drawable.
417      */
setViewDrawable(ImageView v, Drawable drawable, int nullVisibility)418     private void setViewDrawable(ImageView v, Drawable drawable, int nullVisibility) {
419         // Set the icon even if the drawable is null, since we need to clear any
420         // previous icon.
421         v.setImageDrawable(drawable);
422 
423         if (drawable == null) {
424             v.setVisibility(nullVisibility);
425         } else {
426             v.setVisibility(View.VISIBLE);
427 
428             // This is a hack to get any animated drawables (like a 'working' spinner)
429             // to animate. You have to setVisible true on an AnimationDrawable to get
430             // it to start animating, but it must first have been false or else the
431             // call to setVisible will be ineffective. We need to clear up the story
432             // about animated drawables in the future, see http://b/1878430.
433             drawable.setVisible(false, false);
434             drawable.setVisible(true, false);
435         }
436     }
437 
438     /**
439      * Gets the text to show in the query field when a suggestion is selected.
440      *
441      * @param cursor The Cursor to read the suggestion data from. The Cursor should already
442      *        be moved to the suggestion that is to be read from.
443      * @return The text to show, or <code>null</code> if the query should not be
444      *         changed when selecting this suggestion.
445      */
446     @Override
convertToString(Cursor cursor)447     public CharSequence convertToString(Cursor cursor) {
448         if (cursor == null) {
449             return null;
450         }
451 
452         String query = getColumnString(cursor, SearchManager.SUGGEST_COLUMN_QUERY);
453         if (query != null) {
454             return query;
455         }
456 
457         if (mSearchable.shouldRewriteQueryFromData()) {
458             String data = getColumnString(cursor, SearchManager.SUGGEST_COLUMN_INTENT_DATA);
459             if (data != null) {
460                 return data;
461             }
462         }
463 
464         if (mSearchable.shouldRewriteQueryFromText()) {
465             String text1 = getColumnString(cursor, SearchManager.SUGGEST_COLUMN_TEXT_1);
466             if (text1 != null) {
467                 return text1;
468             }
469         }
470 
471         return null;
472     }
473 
474     /**
475      * This method is overridden purely to provide a bit of protection against
476      * flaky content providers.
477      *
478      * @see android.widget.ListAdapter#getView(int, View, ViewGroup)
479      */
480     @Override
getView(int position, View convertView, ViewGroup parent)481     public View getView(int position, View convertView, ViewGroup parent) {
482         try {
483             return super.getView(position, convertView, parent);
484         } catch (RuntimeException e) {
485             Log.w(LOG_TAG, "Search suggestions cursor threw exception.", e);
486             // Put exception string in item title
487             View v = newView(mContext, mCursor, parent);
488             if (v != null) {
489                 ChildViewCache views = (ChildViewCache) v.getTag();
490                 TextView tv = views.mText1;
491                 tv.setText(e.toString());
492             }
493             return v;
494         }
495     }
496 
497     /**
498      * Gets a drawable given a value provided by a suggestion provider.
499      *
500      * This value could be just the string value of a resource id
501      * (e.g., "2130837524"), in which case we will try to retrieve a drawable from
502      * the provider's resources. If the value is not an integer, it is
503      * treated as a Uri and opened with
504      * {@link ContentResolver#openOutputStream(android.net.Uri, String)}.
505      *
506      * All resources and URIs are read using the suggestion provider's context.
507      *
508      * If the string is not formatted as expected, or no drawable can be found for
509      * the provided value, this method returns null.
510      *
511      * @param drawableId a string like "2130837524",
512      *        "android.resource://com.android.alarmclock/2130837524",
513      *        or "content://contacts/photos/253".
514      * @return a Drawable, or null if none found
515      */
getDrawableFromResourceValue(String drawableId)516     private Drawable getDrawableFromResourceValue(String drawableId) {
517         if (drawableId == null || drawableId.length() == 0 || "0".equals(drawableId)) {
518             return null;
519         }
520         try {
521             // First, see if it's just an integer
522             int resourceId = Integer.parseInt(drawableId);
523             // It's an int, look for it in the cache
524             String drawableUri = ContentResolver.SCHEME_ANDROID_RESOURCE
525                     + "://" + mProviderContext.getPackageName() + "/" + resourceId;
526             // Must use URI as cache key, since ints are app-specific
527             Drawable drawable = checkIconCache(drawableUri);
528             if (drawable != null) {
529                 return drawable;
530             }
531             // Not cached, find it by resource ID
532             drawable = mProviderContext.getResources().getDrawable(resourceId);
533             // Stick it in the cache, using the URI as key
534             storeInIconCache(drawableUri, drawable);
535             return drawable;
536         } catch (NumberFormatException nfe) {
537             // It's not an integer, use it as a URI
538             Drawable drawable = checkIconCache(drawableId);
539             if (drawable != null) {
540                 return drawable;
541             }
542             Uri uri = Uri.parse(drawableId);
543             drawable = getDrawable(uri);
544             storeInIconCache(drawableId, drawable);
545             return drawable;
546         } catch (Resources.NotFoundException nfe) {
547             // It was an integer, but it couldn't be found, bail out
548             Log.w(LOG_TAG, "Icon resource not found: " + drawableId);
549             return null;
550         }
551     }
552 
553     /**
554      * Gets a drawable by URI, without using the cache.
555      *
556      * @return A drawable, or {@code null} if the drawable could not be loaded.
557      */
getDrawable(Uri uri)558     private Drawable getDrawable(Uri uri) {
559         try {
560             String scheme = uri.getScheme();
561             if (ContentResolver.SCHEME_ANDROID_RESOURCE.equals(scheme)) {
562                 // Load drawables through Resources, to get the source density information
563                 OpenResourceIdResult r =
564                     mProviderContext.getContentResolver().getResourceId(uri);
565                 try {
566                     return r.r.getDrawable(r.id);
567                 } catch (Resources.NotFoundException ex) {
568                     throw new FileNotFoundException("Resource does not exist: " + uri);
569                 }
570             } else {
571                 // Let the ContentResolver handle content and file URIs.
572                 InputStream stream = mProviderContext.getContentResolver().openInputStream(uri);
573                 if (stream == null) {
574                     throw new FileNotFoundException("Failed to open " + uri);
575                 }
576                 try {
577                     return Drawable.createFromStream(stream, null);
578                 } finally {
579                     try {
580                         stream.close();
581                     } catch (IOException ex) {
582                         Log.e(LOG_TAG, "Error closing icon stream for " + uri, ex);
583                     }
584                 }
585             }
586         } catch (FileNotFoundException fnfe) {
587             Log.w(LOG_TAG, "Icon not found: " + uri + ", " + fnfe.getMessage());
588             return null;
589         }
590     }
591 
checkIconCache(String resourceUri)592     private Drawable checkIconCache(String resourceUri) {
593         Drawable.ConstantState cached = mOutsideDrawablesCache.get(resourceUri);
594         if (cached == null) {
595             return null;
596         }
597         if (DBG) Log.d(LOG_TAG, "Found icon in cache: " + resourceUri);
598         return cached.newDrawable();
599     }
600 
storeInIconCache(String resourceUri, Drawable drawable)601     private void storeInIconCache(String resourceUri, Drawable drawable) {
602         if (drawable != null) {
603             mOutsideDrawablesCache.put(resourceUri, drawable.getConstantState());
604         }
605     }
606 
607     /**
608      * Gets the left-hand side icon that will be used for the current suggestion
609      * if the suggestion contains an icon column but no icon or a broken icon.
610      *
611      * @param cursor A cursor positioned at the current suggestion.
612      * @return A non-null drawable.
613      */
getDefaultIcon1(Cursor cursor)614     private Drawable getDefaultIcon1(Cursor cursor) {
615         // Check the component that gave us the suggestion
616         Drawable drawable = getActivityIconWithCache(mSearchable.getSearchActivity());
617         if (drawable != null) {
618             return drawable;
619         }
620 
621         // Fall back to a default icon
622         return mContext.getPackageManager().getDefaultActivityIcon();
623     }
624 
625     /**
626      * Gets the activity or application icon for an activity.
627      * Uses the local icon cache for fast repeated lookups.
628      *
629      * @param component Name of an activity.
630      * @return A drawable, or {@code null} if neither the activity nor the application
631      *         has an icon set.
632      */
getActivityIconWithCache(ComponentName component)633     private Drawable getActivityIconWithCache(ComponentName component) {
634         // First check the icon cache
635         String componentIconKey = component.flattenToShortString();
636         // Using containsKey() since we also store null values.
637         if (mOutsideDrawablesCache.containsKey(componentIconKey)) {
638             Drawable.ConstantState cached = mOutsideDrawablesCache.get(componentIconKey);
639             return cached == null ? null : cached.newDrawable(mProviderContext.getResources());
640         }
641         // Then try the activity or application icon
642         Drawable drawable = getActivityIcon(component);
643         // Stick it in the cache so we don't do this lookup again.
644         Drawable.ConstantState toCache = drawable == null ? null : drawable.getConstantState();
645         mOutsideDrawablesCache.put(componentIconKey, toCache);
646         return drawable;
647     }
648 
649     /**
650      * Gets the activity or application icon for an activity.
651      *
652      * @param component Name of an activity.
653      * @return A drawable, or {@code null} if neither the acitivy or the application
654      *         have an icon set.
655      */
getActivityIcon(ComponentName component)656     private Drawable getActivityIcon(ComponentName component) {
657         PackageManager pm = mContext.getPackageManager();
658         final ActivityInfo activityInfo;
659         try {
660             activityInfo = pm.getActivityInfo(component, PackageManager.GET_META_DATA);
661         } catch (NameNotFoundException ex) {
662             Log.w(LOG_TAG, ex.toString());
663             return null;
664         }
665         int iconId = activityInfo.getIconResource();
666         if (iconId == 0) return null;
667         String pkg = component.getPackageName();
668         Drawable drawable = pm.getDrawable(pkg, iconId, activityInfo.applicationInfo);
669         if (drawable == null) {
670             Log.w(LOG_TAG, "Invalid icon resource " + iconId + " for "
671                     + component.flattenToShortString());
672             return null;
673         }
674         return drawable;
675     }
676 
677     /**
678      * Gets the value of a string column by name.
679      *
680      * @param cursor Cursor to read the value from.
681      * @param columnName The name of the column to read.
682      * @return The value of the given column, or <code>null</null>
683      *         if the cursor does not contain the given column.
684      */
getColumnString(Cursor cursor, String columnName)685     public static String getColumnString(Cursor cursor, String columnName) {
686         int col = cursor.getColumnIndex(columnName);
687         return getStringOrNull(cursor, col);
688     }
689 
getStringOrNull(Cursor cursor, int col)690     private static String getStringOrNull(Cursor cursor, int col) {
691         if (col == INVALID_INDEX) {
692             return null;
693         }
694         try {
695             return cursor.getString(col);
696         } catch (Exception e) {
697             Log.e(LOG_TAG,
698                     "unexpected error retrieving valid column from cursor, "
699                             + "did the remote process die?", e);
700             return null;
701         }
702     }
703 }
704