blob: 05f5e90e56d74a5e1f753525c50392ef62fb5be7 [file] [log] [blame]
Kenny Root15a4d2f2010-03-11 18:20:12 -08001/*
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
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080017package android.content.pm;
18
Dianne Hackborn6d8dfbd2013-09-23 17:38:51 -070019import android.content.ComponentName;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080020import android.content.IntentFilter;
21import android.graphics.drawable.Drawable;
22import android.os.Parcel;
23import android.os.Parcelable;
Nicolas Prevot88cc3462014-05-14 14:51:48 +010024import android.os.UserHandle;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080025import android.text.TextUtils;
26import android.util.Printer;
Jeff Sharkey85f5f812013-10-07 10:16:12 -070027import android.util.Slog;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080028
29import java.text.Collator;
30import java.util.Comparator;
31
32/**
33 * Information that is returned from resolving an intent
34 * against an IntentFilter. This partially corresponds to
35 * information collected from the AndroidManifest.xml's
36 * <intent> tags.
37 */
38public class ResolveInfo implements Parcelable {
Jeff Sharkey85f5f812013-10-07 10:16:12 -070039 private static final String TAG = "ResolveInfo";
40
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080041 /**
Jeff Sharkey85f5f812013-10-07 10:16:12 -070042 * The activity or broadcast receiver that corresponds to this resolution
43 * match, if this resolution is for an activity or broadcast receiver.
44 * Exactly one of {@link #activityInfo}, {@link #serviceInfo}, or
45 * {@link #providerInfo} will be non-null.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080046 */
47 public ActivityInfo activityInfo;
48
49 /**
Jeff Sharkey85f5f812013-10-07 10:16:12 -070050 * The service that corresponds to this resolution match, if this resolution
51 * is for a service. Exactly one of {@link #activityInfo},
52 * {@link #serviceInfo}, or {@link #providerInfo} will be non-null.
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080053 */
54 public ServiceInfo serviceInfo;
Jeff Sharkey85f5f812013-10-07 10:16:12 -070055
56 /**
57 * The provider that corresponds to this resolution match, if this
58 * resolution is for a provider. Exactly one of {@link #activityInfo},
59 * {@link #serviceInfo}, or {@link #providerInfo} will be non-null.
60 */
61 public ProviderInfo providerInfo;
62
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080063 /**
64 * The IntentFilter that was matched for this ResolveInfo.
65 */
66 public IntentFilter filter;
67
68 /**
69 * The declared priority of this match. Comes from the "priority"
70 * attribute or, if not set, defaults to 0. Higher values are a higher
71 * priority.
72 */
73 public int priority;
74
75 /**
76 * Order of result according to the user's preference. If the user
77 * has not set a preference for this result, the value is 0; higher
78 * values are a higher priority.
79 */
80 public int preferredOrder;
81
82 /**
83 * The system's evaluation of how well the activity matches the
84 * IntentFilter. This is a match constant, a combination of
85 * {@link IntentFilter#MATCH_CATEGORY_MASK IntentFilter.MATCH_CATEGORY_MASK}
86 * and {@link IntentFilter#MATCH_ADJUSTMENT_MASK IntentFiler.MATCH_ADJUSTMENT_MASK}.
87 */
88 public int match;
89
90 /**
91 * Only set when returned by
92 * {@link PackageManager#queryIntentActivityOptions}, this tells you
93 * which of the given specific intents this result came from. 0 is the
94 * first in the list, < 0 means it came from the generic Intent query.
95 */
96 public int specificIndex = -1;
97
98 /**
99 * This filter has specified the Intent.CATEGORY_DEFAULT, meaning it
100 * would like to be considered a default action that the user can
101 * perform on this data.
102 */
103 public boolean isDefault;
104
105 /**
106 * A string resource identifier (in the package's resources) of this
107 * match's label. From the "label" attribute or, if not set, 0.
108 */
109 public int labelRes;
110
111 /**
112 * The actual string retrieve from <var>labelRes</var> or null if none
113 * was provided.
114 */
115 public CharSequence nonLocalizedLabel;
116
117 /**
118 * A drawable resource identifier (in the package's resources) of this
119 * match's icon. From the "icon" attribute or, if not set, 0.
120 */
121 public int icon;
122
123 /**
Dianne Hackborneb034652009-09-07 00:49:58 -0700124 * Optional -- if non-null, the {@link #labelRes} and {@link #icon}
125 * resources will be loaded from this package, rather than the one
126 * containing the resolved component.
127 */
128 public String resolvePackageName;
Dianne Hackbornd99b2932011-08-18 14:39:58 -0700129
130 /**
Nicolas Prevot88cc3462014-05-14 14:51:48 +0100131 * If not equal to UserHandle.USER_CURRENT, then the intent will be forwarded to this user.
132 * @hide
133 */
134 public int targetUserId;
135
136 /**
Nicolas Prevot88cc3462014-05-14 14:51:48 +0100137 * @hide
138 */
Nicolas Prevot7f7b0c72014-06-23 15:59:38 +0100139 public boolean noResourceId;
Nicolas Prevot88cc3462014-05-14 14:51:48 +0100140
141 /**
Dianne Hackbornd99b2932011-08-18 14:39:58 -0700142 * @hide Target comes from system process?
143 */
144 public boolean system;
145
Fabrice Di Meglio1c1b4712014-11-19 17:12:32 -0800146 /**
Fabrice Di Meglio7d014ce2015-04-08 16:17:46 -0700147 * @hide Does the associated IntentFilter comes from a Browser ?
Fabrice Di Meglio1c1b4712014-11-19 17:12:32 -0800148 */
Fabrice Di Meglio7d014ce2015-04-08 16:17:46 -0700149 public boolean handleAllWebDataURI;
Fabrice Di Meglio1c1b4712014-11-19 17:12:32 -0800150
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700151 private ComponentInfo getComponentInfo() {
152 if (activityInfo != null) return activityInfo;
153 if (serviceInfo != null) return serviceInfo;
154 if (providerInfo != null) return providerInfo;
155 throw new IllegalStateException("Missing ComponentInfo!");
156 }
157
Dianne Hackborneb034652009-09-07 00:49:58 -0700158 /**
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800159 * Retrieve the current textual label associated with this resolution. This
160 * will call back on the given PackageManager to load the label from
161 * the application.
162 *
163 * @param pm A PackageManager from which the label can be loaded; usually
164 * the PackageManager from which you originally retrieved this item.
165 *
166 * @return Returns a CharSequence containing the resolutions's label. If the
167 * item does not have a label, its name is returned.
168 */
169 public CharSequence loadLabel(PackageManager pm) {
170 if (nonLocalizedLabel != null) {
171 return nonLocalizedLabel;
172 }
Dianne Hackborneb034652009-09-07 00:49:58 -0700173 CharSequence label;
174 if (resolvePackageName != null && labelRes != 0) {
175 label = pm.getText(resolvePackageName, labelRes, null);
176 if (label != null) {
Romain Guy2aba11f2010-03-29 16:03:01 -0700177 return label.toString().trim();
Dianne Hackborneb034652009-09-07 00:49:58 -0700178 }
179 }
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700180 ComponentInfo ci = getComponentInfo();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800181 ApplicationInfo ai = ci.applicationInfo;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800182 if (labelRes != 0) {
183 label = pm.getText(ci.packageName, labelRes, ai);
184 if (label != null) {
Romain Guy2aba11f2010-03-29 16:03:01 -0700185 return label.toString().trim();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800186 }
187 }
Romain Guy2aba11f2010-03-29 16:03:01 -0700188
189 CharSequence data = ci.loadLabel(pm);
190 // Make the data safe
191 if (data != null) data = data.toString().trim();
192 return data;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800193 }
194
195 /**
196 * Retrieve the current graphical icon associated with this resolution. This
197 * will call back on the given PackageManager to load the icon from
198 * the application.
199 *
200 * @param pm A PackageManager from which the icon can be loaded; usually
201 * the PackageManager from which you originally retrieved this item.
202 *
203 * @return Returns a Drawable containing the resolution's icon. If the
204 * item does not have an icon, the default activity icon is returned.
205 */
206 public Drawable loadIcon(PackageManager pm) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800207 Drawable dr;
Dianne Hackborneb034652009-09-07 00:49:58 -0700208 if (resolvePackageName != null && icon != 0) {
209 dr = pm.getDrawable(resolvePackageName, icon, null);
210 if (dr != null) {
211 return dr;
212 }
213 }
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700214 ComponentInfo ci = getComponentInfo();
Jeff Brown07330792010-03-30 19:57:08 -0700215 ApplicationInfo ai = ci.applicationInfo;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800216 if (icon != 0) {
217 dr = pm.getDrawable(ci.packageName, icon, ai);
218 if (dr != null) {
219 return dr;
220 }
221 }
222 return ci.loadIcon(pm);
223 }
224
225 /**
226 * Return the icon resource identifier to use for this match. If the
227 * match defines an icon, that is used; else if the activity defines
228 * an icon, that is used; else, the application icon is used.
229 *
230 * @return The icon associated with this match.
231 */
232 public final int getIconResource() {
Nicolas Prevot7f7b0c72014-06-23 15:59:38 +0100233 if (noResourceId) return 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800234 if (icon != 0) return icon;
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700235 final ComponentInfo ci = getComponentInfo();
Nicolas Prevot7f7b0c72014-06-23 15:59:38 +0100236 if (ci != null) {
Nicolas Prevot88cc3462014-05-14 14:51:48 +0100237 return ci.getIconResource();
238 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800239 return 0;
240 }
241
242 public void dump(Printer pw, String prefix) {
243 if (filter != null) {
244 pw.println(prefix + "Filter:");
245 filter.dump(pw, prefix + " ");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800246 }
247 pw.println(prefix + "priority=" + priority
248 + " preferredOrder=" + preferredOrder
249 + " match=0x" + Integer.toHexString(match)
250 + " specificIndex=" + specificIndex
251 + " isDefault=" + isDefault);
Dianne Hackborneb034652009-09-07 00:49:58 -0700252 if (resolvePackageName != null) {
253 pw.println(prefix + "resolvePackageName=" + resolvePackageName);
254 }
255 if (labelRes != 0 || nonLocalizedLabel != null || icon != 0) {
256 pw.println(prefix + "labelRes=0x" + Integer.toHexString(labelRes)
257 + " nonLocalizedLabel=" + nonLocalizedLabel
258 + " icon=0x" + Integer.toHexString(icon));
259 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800260 if (activityInfo != null) {
261 pw.println(prefix + "ActivityInfo:");
262 activityInfo.dump(pw, prefix + " ");
263 } else if (serviceInfo != null) {
264 pw.println(prefix + "ServiceInfo:");
Dianne Hackborneb034652009-09-07 00:49:58 -0700265 serviceInfo.dump(pw, prefix + " ");
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700266 } else if (providerInfo != null) {
267 pw.println(prefix + "ProviderInfo:");
268 providerInfo.dump(pw, prefix + " ");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800269 }
270 }
271
272 public ResolveInfo() {
Nicolas Prevot88cc3462014-05-14 14:51:48 +0100273 targetUserId = UserHandle.USER_CURRENT;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800274 }
275
Dianne Hackborn8da429e2012-09-23 12:52:19 -0700276 public ResolveInfo(ResolveInfo orig) {
277 activityInfo = orig.activityInfo;
278 serviceInfo = orig.serviceInfo;
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700279 providerInfo = orig.providerInfo;
Dianne Hackborn8da429e2012-09-23 12:52:19 -0700280 filter = orig.filter;
281 priority = orig.priority;
282 preferredOrder = orig.preferredOrder;
283 match = orig.match;
284 specificIndex = orig.specificIndex;
285 labelRes = orig.labelRes;
286 nonLocalizedLabel = orig.nonLocalizedLabel;
287 icon = orig.icon;
288 resolvePackageName = orig.resolvePackageName;
289 system = orig.system;
Nicolas Prevot88cc3462014-05-14 14:51:48 +0100290 targetUserId = orig.targetUserId;
Fabrice Di Meglio7d014ce2015-04-08 16:17:46 -0700291 handleAllWebDataURI = orig.handleAllWebDataURI;
Dianne Hackborn8da429e2012-09-23 12:52:19 -0700292 }
293
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800294 public String toString() {
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700295 final ComponentInfo ci = getComponentInfo();
Dianne Hackborn6d8dfbd2013-09-23 17:38:51 -0700296 StringBuilder sb = new StringBuilder(128);
297 sb.append("ResolveInfo{");
298 sb.append(Integer.toHexString(System.identityHashCode(this)));
299 sb.append(' ');
300 ComponentName.appendShortString(sb, ci.packageName, ci.name);
301 if (priority != 0) {
302 sb.append(" p=");
303 sb.append(priority);
304 }
305 if (preferredOrder != 0) {
306 sb.append(" o=");
307 sb.append(preferredOrder);
308 }
309 sb.append(" m=0x");
310 sb.append(Integer.toHexString(match));
Nicolas Prevot88cc3462014-05-14 14:51:48 +0100311 if (targetUserId != UserHandle.USER_CURRENT) {
312 sb.append(" targetUserId=");
313 sb.append(targetUserId);
314 }
Dianne Hackborn6d8dfbd2013-09-23 17:38:51 -0700315 sb.append('}');
316 return sb.toString();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800317 }
318
319 public int describeContents() {
320 return 0;
321 }
322
323 public void writeToParcel(Parcel dest, int parcelableFlags) {
324 if (activityInfo != null) {
325 dest.writeInt(1);
326 activityInfo.writeToParcel(dest, parcelableFlags);
327 } else if (serviceInfo != null) {
328 dest.writeInt(2);
329 serviceInfo.writeToParcel(dest, parcelableFlags);
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700330 } else if (providerInfo != null) {
331 dest.writeInt(3);
332 providerInfo.writeToParcel(dest, parcelableFlags);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800333 } else {
334 dest.writeInt(0);
335 }
336 if (filter != null) {
337 dest.writeInt(1);
338 filter.writeToParcel(dest, parcelableFlags);
339 } else {
340 dest.writeInt(0);
341 }
342 dest.writeInt(priority);
343 dest.writeInt(preferredOrder);
344 dest.writeInt(match);
345 dest.writeInt(specificIndex);
346 dest.writeInt(labelRes);
347 TextUtils.writeToParcel(nonLocalizedLabel, dest, parcelableFlags);
348 dest.writeInt(icon);
Dianne Hackborneb034652009-09-07 00:49:58 -0700349 dest.writeString(resolvePackageName);
Nicolas Prevot88cc3462014-05-14 14:51:48 +0100350 dest.writeInt(targetUserId);
Dianne Hackbornd99b2932011-08-18 14:39:58 -0700351 dest.writeInt(system ? 1 : 0);
Nicolas Prevot7f7b0c72014-06-23 15:59:38 +0100352 dest.writeInt(noResourceId ? 1 : 0);
Fabrice Di Meglio7d014ce2015-04-08 16:17:46 -0700353 dest.writeInt(handleAllWebDataURI ? 1 : 0);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800354 }
355
356 public static final Creator<ResolveInfo> CREATOR
357 = new Creator<ResolveInfo>() {
358 public ResolveInfo createFromParcel(Parcel source) {
359 return new ResolveInfo(source);
360 }
361 public ResolveInfo[] newArray(int size) {
362 return new ResolveInfo[size];
363 }
364 };
365
366 private ResolveInfo(Parcel source) {
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700367 activityInfo = null;
368 serviceInfo = null;
369 providerInfo = null;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800370 switch (source.readInt()) {
371 case 1:
372 activityInfo = ActivityInfo.CREATOR.createFromParcel(source);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800373 break;
374 case 2:
375 serviceInfo = ServiceInfo.CREATOR.createFromParcel(source);
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700376 break;
377 case 3:
378 providerInfo = ProviderInfo.CREATOR.createFromParcel(source);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800379 break;
380 default:
Jeff Sharkey85f5f812013-10-07 10:16:12 -0700381 Slog.w(TAG, "Missing ComponentInfo!");
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800382 break;
383 }
384 if (source.readInt() != 0) {
385 filter = IntentFilter.CREATOR.createFromParcel(source);
386 }
387 priority = source.readInt();
388 preferredOrder = source.readInt();
389 match = source.readInt();
390 specificIndex = source.readInt();
391 labelRes = source.readInt();
392 nonLocalizedLabel
393 = TextUtils.CHAR_SEQUENCE_CREATOR.createFromParcel(source);
394 icon = source.readInt();
Dianne Hackborneb034652009-09-07 00:49:58 -0700395 resolvePackageName = source.readString();
Nicolas Prevot88cc3462014-05-14 14:51:48 +0100396 targetUserId = source.readInt();
Dianne Hackbornd99b2932011-08-18 14:39:58 -0700397 system = source.readInt() != 0;
Nicolas Prevot7f7b0c72014-06-23 15:59:38 +0100398 noResourceId = source.readInt() != 0;
Fabrice Di Meglio7d014ce2015-04-08 16:17:46 -0700399 handleAllWebDataURI = source.readInt() != 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800400 }
401
402 public static class DisplayNameComparator
403 implements Comparator<ResolveInfo> {
404 public DisplayNameComparator(PackageManager pm) {
405 mPM = pm;
Adam Powell0256c6f2013-05-29 16:42:33 -0700406 mCollator.setStrength(Collator.PRIMARY);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800407 }
408
409 public final int compare(ResolveInfo a, ResolveInfo b) {
Nicolas Prevot88cc3462014-05-14 14:51:48 +0100410 // We want to put the one targeted to another user at the end of the dialog.
411 if (a.targetUserId != UserHandle.USER_CURRENT) {
412 return 1;
413 }
414 if (b.targetUserId != UserHandle.USER_CURRENT) {
415 return -1;
416 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800417 CharSequence sa = a.loadLabel(mPM);
418 if (sa == null) sa = a.activityInfo.name;
419 CharSequence sb = b.loadLabel(mPM);
420 if (sb == null) sb = b.activityInfo.name;
421
Adam Powell0256c6f2013-05-29 16:42:33 -0700422 return mCollator.compare(sa.toString(), sb.toString());
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800423 }
424
Adam Powell0256c6f2013-05-29 16:42:33 -0700425 private final Collator mCollator = Collator.getInstance();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800426 private PackageManager mPM;
427 }
428}