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.app;
18 
19 import android.content.Context;
20 import android.content.Intent;
21 import android.content.pm.ComponentInfo;
22 import android.content.pm.PackageManager;
23 import android.content.pm.ResolveInfo;
24 import android.content.res.Resources;
25 import android.graphics.Bitmap;
26 import android.graphics.Canvas;
27 import android.graphics.Paint;
28 import android.graphics.PaintFlagsDrawFilter;
29 import android.graphics.PixelFormat;
30 import android.graphics.Rect;
31 import android.graphics.drawable.BitmapDrawable;
32 import android.graphics.drawable.Drawable;
33 import android.graphics.drawable.PaintDrawable;
34 import android.os.Bundle;
35 import android.view.LayoutInflater;
36 import android.view.View;
37 import android.view.ViewGroup;
38 import android.view.Window;
39 import android.view.View.OnClickListener;
40 import android.widget.BaseAdapter;
41 import android.widget.Button;
42 import android.widget.Filter;
43 import android.widget.Filterable;
44 import android.widget.ListView;
45 import android.widget.TextView;
46 
47 import java.util.ArrayList;
48 import java.util.Collections;
49 import java.util.List;
50 
51 
52 /**
53  * Displays a list of all activities which can be performed
54  * for a given intent. Launches when clicked.
55  *
56  */
57 public abstract class LauncherActivity extends ListActivity {
58     Intent mIntent;
59     PackageManager mPackageManager;
60     IconResizer mIconResizer;
61 
62     /**
63      * An item in the list
64      */
65     public static class ListItem {
66         public ResolveInfo resolveInfo;
67         public CharSequence label;
68         public Drawable icon;
69         public String packageName;
70         public String className;
71         public Bundle extras;
72 
ListItem(PackageManager pm, ResolveInfo resolveInfo, IconResizer resizer)73         ListItem(PackageManager pm, ResolveInfo resolveInfo, IconResizer resizer) {
74             this.resolveInfo = resolveInfo;
75             label = resolveInfo.loadLabel(pm);
76             ComponentInfo ci = resolveInfo.activityInfo;
77             if (ci == null) ci = resolveInfo.serviceInfo;
78             if (label == null && ci != null) {
79                 label = resolveInfo.activityInfo.name;
80             }
81 
82             if (resizer != null) {
83                 icon = resizer.createIconThumbnail(resolveInfo.loadIcon(pm));
84             }
85             packageName = ci.applicationInfo.packageName;
86             className = ci.name;
87         }
88 
ListItem()89         public ListItem() {
90         }
91     }
92 
93     /**
94      * Adapter which shows the set of activities that can be performed for a given intent.
95      */
96     private class ActivityAdapter extends BaseAdapter implements Filterable {
97         private final Object lock = new Object();
98         private ArrayList<ListItem> mOriginalValues;
99 
100         protected final IconResizer mIconResizer;
101         protected final LayoutInflater mInflater;
102 
103         protected List<ListItem> mActivitiesList;
104 
105         private Filter mFilter;
106         private final boolean mShowIcons;
107 
ActivityAdapter(IconResizer resizer)108         public ActivityAdapter(IconResizer resizer) {
109             mIconResizer = resizer;
110             mInflater = (LayoutInflater) LauncherActivity.this.getSystemService(
111                     Context.LAYOUT_INFLATER_SERVICE);
112             mShowIcons = onEvaluateShowIcons();
113             mActivitiesList = makeListItems();
114         }
115 
intentForPosition(int position)116         public Intent intentForPosition(int position) {
117             if (mActivitiesList == null) {
118                 return null;
119             }
120 
121             Intent intent = new Intent(mIntent);
122             ListItem item = mActivitiesList.get(position);
123             intent.setClassName(item.packageName, item.className);
124             if (item.extras != null) {
125                 intent.putExtras(item.extras);
126             }
127             return intent;
128         }
129 
itemForPosition(int position)130         public ListItem itemForPosition(int position) {
131             if (mActivitiesList == null) {
132                 return null;
133             }
134 
135             return mActivitiesList.get(position);
136         }
137 
getCount()138         public int getCount() {
139             return mActivitiesList != null ? mActivitiesList.size() : 0;
140         }
141 
getItem(int position)142         public Object getItem(int position) {
143             return position;
144         }
145 
getItemId(int position)146         public long getItemId(int position) {
147             return position;
148         }
149 
getView(int position, View convertView, ViewGroup parent)150         public View getView(int position, View convertView, ViewGroup parent) {
151             View view;
152             if (convertView == null) {
153                 view = mInflater.inflate(
154                         com.android.internal.R.layout.activity_list_item_2, parent, false);
155             } else {
156                 view = convertView;
157             }
158             bindView(view, mActivitiesList.get(position));
159             return view;
160         }
161 
bindView(View view, ListItem item)162         private void bindView(View view, ListItem item) {
163             TextView text = (TextView) view;
164             text.setText(item.label);
165             if (mShowIcons) {
166                 if (item.icon == null) {
167                     item.icon = mIconResizer.createIconThumbnail(item.resolveInfo.loadIcon(getPackageManager()));
168                 }
169                 text.setCompoundDrawablesWithIntrinsicBounds(item.icon, null, null, null);
170             }
171         }
172 
getFilter()173         public Filter getFilter() {
174             if (mFilter == null) {
175                 mFilter = new ArrayFilter();
176             }
177             return mFilter;
178         }
179 
180         /**
181          * An array filters constrains the content of the array adapter with a prefix. Each
182          * item that does not start with the supplied prefix is removed from the list.
183          */
184         private class ArrayFilter extends Filter {
185             @Override
performFiltering(CharSequence prefix)186             protected FilterResults performFiltering(CharSequence prefix) {
187                 FilterResults results = new FilterResults();
188 
189                 if (mOriginalValues == null) {
190                     synchronized (lock) {
191                         mOriginalValues = new ArrayList<ListItem>(mActivitiesList);
192                     }
193                 }
194 
195                 if (prefix == null || prefix.length() == 0) {
196                     synchronized (lock) {
197                         ArrayList<ListItem> list = new ArrayList<ListItem>(mOriginalValues);
198                         results.values = list;
199                         results.count = list.size();
200                     }
201                 } else {
202                     final String prefixString = prefix.toString().toLowerCase();
203 
204                     ArrayList<ListItem> values = mOriginalValues;
205                     int count = values.size();
206 
207                     ArrayList<ListItem> newValues = new ArrayList<ListItem>(count);
208 
209                     for (int i = 0; i < count; i++) {
210                         ListItem item = values.get(i);
211 
212                         String[] words = item.label.toString().toLowerCase().split(" ");
213                         int wordCount = words.length;
214 
215                         for (int k = 0; k < wordCount; k++) {
216                             final String word = words[k];
217 
218                             if (word.startsWith(prefixString)) {
219                                 newValues.add(item);
220                                 break;
221                             }
222                         }
223                     }
224 
225                     results.values = newValues;
226                     results.count = newValues.size();
227                 }
228 
229                 return results;
230             }
231 
232             @Override
publishResults(CharSequence constraint, FilterResults results)233             protected void publishResults(CharSequence constraint, FilterResults results) {
234                 //noinspection unchecked
235                 mActivitiesList = (List<ListItem>) results.values;
236                 if (results.count > 0) {
237                     notifyDataSetChanged();
238                 } else {
239                     notifyDataSetInvalidated();
240                 }
241             }
242         }
243     }
244 
245     /**
246      * Utility class to resize icons to match default icon size.
247      */
248     public class IconResizer {
249         // Code is borrowed from com.android.launcher.Utilities.
250         private int mIconWidth = -1;
251         private int mIconHeight = -1;
252 
253         private final Rect mOldBounds = new Rect();
254         private Canvas mCanvas = new Canvas();
255 
IconResizer()256         public IconResizer() {
257             mCanvas.setDrawFilter(new PaintFlagsDrawFilter(Paint.DITHER_FLAG,
258                     Paint.FILTER_BITMAP_FLAG));
259 
260             final Resources resources = LauncherActivity.this.getResources();
261             mIconWidth = mIconHeight = (int) resources.getDimension(
262                     android.R.dimen.app_icon_size);
263         }
264 
265         /**
266          * Returns a Drawable representing the thumbnail of the specified Drawable.
267          * The size of the thumbnail is defined by the dimension
268          * android.R.dimen.launcher_application_icon_size.
269          *
270          * This method is not thread-safe and should be invoked on the UI thread only.
271          *
272          * @param icon The icon to get a thumbnail of.
273          *
274          * @return A thumbnail for the specified icon or the icon itself if the
275          *         thumbnail could not be created.
276          */
createIconThumbnail(Drawable icon)277         public Drawable createIconThumbnail(Drawable icon) {
278             int width = mIconWidth;
279             int height = mIconHeight;
280 
281             final int iconWidth = icon.getIntrinsicWidth();
282             final int iconHeight = icon.getIntrinsicHeight();
283 
284             if (icon instanceof PaintDrawable) {
285                 PaintDrawable painter = (PaintDrawable) icon;
286                 painter.setIntrinsicWidth(width);
287                 painter.setIntrinsicHeight(height);
288             }
289 
290             if (width > 0 && height > 0) {
291                 if (width < iconWidth || height < iconHeight) {
292                     final float ratio = (float) iconWidth / iconHeight;
293 
294                     if (iconWidth > iconHeight) {
295                         height = (int) (width / ratio);
296                     } else if (iconHeight > iconWidth) {
297                         width = (int) (height * ratio);
298                     }
299 
300                     final Bitmap.Config c = icon.getOpacity() != PixelFormat.OPAQUE ?
301                                 Bitmap.Config.ARGB_8888 : Bitmap.Config.RGB_565;
302                     final Bitmap thumb = Bitmap.createBitmap(mIconWidth, mIconHeight, c);
303                     final Canvas canvas = mCanvas;
304                     canvas.setBitmap(thumb);
305                     // Copy the old bounds to restore them later
306                     // If we were to do oldBounds = icon.getBounds(),
307                     // the call to setBounds() that follows would
308                     // change the same instance and we would lose the
309                     // old bounds
310                     mOldBounds.set(icon.getBounds());
311                     final int x = (mIconWidth - width) / 2;
312                     final int y = (mIconHeight - height) / 2;
313                     icon.setBounds(x, y, x + width, y + height);
314                     icon.draw(canvas);
315                     icon.setBounds(mOldBounds);
316                     icon = new BitmapDrawable(getResources(), thumb);
317                     canvas.setBitmap(null);
318                 } else if (iconWidth < width && iconHeight < height) {
319                     final Bitmap.Config c = Bitmap.Config.ARGB_8888;
320                     final Bitmap thumb = Bitmap.createBitmap(mIconWidth, mIconHeight, c);
321                     final Canvas canvas = mCanvas;
322                     canvas.setBitmap(thumb);
323                     mOldBounds.set(icon.getBounds());
324                     final int x = (width - iconWidth) / 2;
325                     final int y = (height - iconHeight) / 2;
326                     icon.setBounds(x, y, x + iconWidth, y + iconHeight);
327                     icon.draw(canvas);
328                     icon.setBounds(mOldBounds);
329                     icon = new BitmapDrawable(getResources(), thumb);
330                     canvas.setBitmap(null);
331                 }
332             }
333 
334             return icon;
335         }
336     }
337 
338     @Override
onCreate(Bundle icicle)339     protected void onCreate(Bundle icicle) {
340         super.onCreate(icicle);
341 
342         mPackageManager = getPackageManager();
343 
344         if (!mPackageManager.hasSystemFeature(PackageManager.FEATURE_WATCH)) {
345             requestWindowFeature(Window.FEATURE_INDETERMINATE_PROGRESS);
346             setProgressBarIndeterminateVisibility(true);
347         }
348         onSetContentView();
349 
350         mIconResizer = new IconResizer();
351 
352         mIntent = new Intent(getTargetIntent());
353         mIntent.setComponent(null);
354         mAdapter = new ActivityAdapter(mIconResizer);
355 
356         setListAdapter(mAdapter);
357         getListView().setTextFilterEnabled(true);
358 
359         updateAlertTitle();
360         updateButtonText();
361 
362         if (!mPackageManager.hasSystemFeature(PackageManager.FEATURE_WATCH)) {
363             setProgressBarIndeterminateVisibility(false);
364         }
365     }
366 
updateAlertTitle()367     private void updateAlertTitle() {
368         TextView alertTitle = (TextView) findViewById(com.android.internal.R.id.alertTitle);
369         if (alertTitle != null) {
370             alertTitle.setText(getTitle());
371         }
372     }
373 
updateButtonText()374     private void updateButtonText() {
375         Button cancelButton = (Button) findViewById(com.android.internal.R.id.button1);
376         if (cancelButton != null) {
377             cancelButton.setOnClickListener(new OnClickListener() {
378                 public void onClick(View v) {
379                     finish();
380                 }
381             });
382         }
383     }
384 
385     @Override
setTitle(CharSequence title)386     public void setTitle(CharSequence title) {
387         super.setTitle(title);
388         updateAlertTitle();
389     }
390 
391     @Override
setTitle(int titleId)392     public void setTitle(int titleId) {
393         super.setTitle(titleId);
394         updateAlertTitle();
395     }
396 
397     /**
398      * Override to call setContentView() with your own content view to
399      * customize the list layout.
400      */
onSetContentView()401     protected void onSetContentView() {
402         setContentView(com.android.internal.R.layout.activity_list);
403     }
404 
405     @Override
onListItemClick(ListView l, View v, int position, long id)406     protected void onListItemClick(ListView l, View v, int position, long id) {
407         Intent intent = intentForPosition(position);
408         startActivity(intent);
409     }
410 
411     /**
412      * Return the actual Intent for a specific position in our
413      * {@link android.widget.ListView}.
414      * @param position The item whose Intent to return
415      */
intentForPosition(int position)416     protected Intent intentForPosition(int position) {
417         ActivityAdapter adapter = (ActivityAdapter) mAdapter;
418         return adapter.intentForPosition(position);
419     }
420 
421     /**
422      * Return the {@link ListItem} for a specific position in our
423      * {@link android.widget.ListView}.
424      * @param position The item to return
425      */
itemForPosition(int position)426     protected ListItem itemForPosition(int position) {
427         ActivityAdapter adapter = (ActivityAdapter) mAdapter;
428         return adapter.itemForPosition(position);
429     }
430 
431     /**
432      * Get the base intent to use when running
433      * {@link PackageManager#queryIntentActivities(Intent, int)}.
434      */
getTargetIntent()435     protected Intent getTargetIntent() {
436         return new Intent();
437     }
438 
439     /**
440      * Perform query on package manager for list items.  The default
441      * implementation queries for activities.
442      */
onQueryPackageManager(Intent queryIntent)443     protected List<ResolveInfo> onQueryPackageManager(Intent queryIntent) {
444         return mPackageManager.queryIntentActivities(queryIntent, /* no flags */ 0);
445     }
446 
447     /**
448      * @hide
449      */
onSortResultList(List<ResolveInfo> results)450     protected void onSortResultList(List<ResolveInfo> results) {
451         Collections.sort(results, new ResolveInfo.DisplayNameComparator(mPackageManager));
452     }
453 
454     /**
455      * Perform the query to determine which results to show and return a list of them.
456      */
makeListItems()457     public List<ListItem> makeListItems() {
458         // Load all matching activities and sort correctly
459         List<ResolveInfo> list = onQueryPackageManager(mIntent);
460         onSortResultList(list);
461 
462         ArrayList<ListItem> result = new ArrayList<ListItem>(list.size());
463         int listSize = list.size();
464         for (int i = 0; i < listSize; i++) {
465             ResolveInfo resolveInfo = list.get(i);
466             result.add(new ListItem(mPackageManager, resolveInfo, null));
467         }
468 
469         return result;
470     }
471 
472     /**
473      * Whether or not to show icons in the list
474      * @hide keeping this private for now, since only Settings needs it
475      * @return true to show icons beside the activity names, false otherwise
476      */
onEvaluateShowIcons()477     protected boolean onEvaluateShowIcons() {
478         return true;
479     }
480 }
481