/*
 * Copyright (C) 2007 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package android.content.pm;

import static java.lang.annotation.RetentionPolicy.SOURCE;

import android.annotation.FloatRange;
import android.annotation.IntDef;
import android.annotation.NonNull;
import android.annotation.SystemApi;
import android.content.res.XmlResourceParser;
import android.graphics.drawable.Drawable;
import android.os.Bundle;
import android.os.Parcel;
import android.os.UserHandle;
import android.text.Html;
import android.text.TextPaint;
import android.text.TextUtils;
import android.util.Printer;
import android.util.proto.ProtoOutputStream;

import com.android.internal.util.Preconditions;

import java.lang.annotation.Retention;
import java.text.Collator;
import java.util.BitSet;
import java.util.Comparator;

Base class containing information common to all package items held by the package manager. This provides a very common basic set of attributes: a label, icon, and meta-data. This class is not intended to be used by itself; it is simply here to share common definitions between all items returned by the package manager. As such, it does not itself implement Parcelable, but does provide convenience methods to assist in the implementation of Parcelable in subclasses.
/** * Base class containing information common to all package items held by * the package manager. This provides a very common basic set of attributes: * a label, icon, and meta-data. This class is not intended * to be used by itself; it is simply here to share common definitions * between all items returned by the package manager. As such, it does not * itself implement Parcelable, but does provide convenience methods to assist * in the implementation of Parcelable in subclasses. */
public class PackageItemInfo { private static final int LINE_FEED_CODE_POINT = 10; private static final int NBSP_CODE_POINT = 160;
@hide
/** * Flags for {@link #loadSafeLabel(PackageManager, float, int)} * * @hide */
@Retention(SOURCE) @IntDef(flag = true, prefix = "SAFE_LABEL_FLAG_", value = {SAFE_LABEL_FLAG_TRIM, SAFE_LABEL_FLAG_SINGLE_LINE, SAFE_LABEL_FLAG_FIRST_LINE}) public @interface SafeLabelFlags {}
Remove whitespace and non-breaking spaces from the edges of the label.
See Also:
@hide
/** * Remove {@link Character#isWhitespace(int) whitespace} and non-breaking spaces from the edges * of the label. * * @see #loadSafeLabel(PackageManager, float, int) * @hide */
public static final int SAFE_LABEL_FLAG_TRIM = 0x1;
Force entire string into single line of text (no newlines). Cannot be set at the same time as SAFE_LABEL_FLAG_FIRST_LINE.
See Also:
@hide
/** * Force entire string into single line of text (no newlines). Cannot be set at the same time as * {@link #SAFE_LABEL_FLAG_FIRST_LINE}. * * @see #loadSafeLabel(PackageManager, float, int) * @hide */
public static final int SAFE_LABEL_FLAG_SINGLE_LINE = 0x2;
Return only first line of text (truncate at first newline). Cannot be set at the same time as SAFE_LABEL_FLAG_SINGLE_LINE.
See Also:
@hide
/** * Return only first line of text (truncate at first newline). Cannot be set at the same time as * {@link #SAFE_LABEL_FLAG_SINGLE_LINE}. * * @see #loadSafeLabel(PackageManager, float, int) * @hide */
public static final int SAFE_LABEL_FLAG_FIRST_LINE = 0x4; private static final float MAX_LABEL_SIZE_PX = 500f;
The maximum length of a safe label, in characters
/** The maximum length of a safe label, in characters */
private static final int MAX_SAFE_LABEL_LENGTH = 50000; private static volatile boolean sForceSafeLabels = false;
{@hide}
/** {@hide} */
public static void setForceSafeLabels(boolean forceSafeLabels) { sForceSafeLabels = forceSafeLabels; }
Public name of this item. From the "android:name" attribute.
/** * Public name of this item. From the "android:name" attribute. */
public String name;
Name of the package that this item is in.
/** * Name of the package that this item is in. */
public String packageName;
A string resource identifier (in the package's resources) of this component's label. From the "label" attribute or, if not set, 0.
/** * A string resource identifier (in the package's resources) of this * component's label. From the "label" attribute or, if not set, 0. */
public int labelRes;
The string provided in the AndroidManifest file, if any. You probably don't want to use this. You probably want PackageManager.getApplicationLabel
/** * The string provided in the AndroidManifest file, if any. You * probably don't want to use this. You probably want * {@link PackageManager#getApplicationLabel} */
public CharSequence nonLocalizedLabel;
A drawable resource identifier (in the package's resources) of this component's icon. From the "icon" attribute or, if not set, 0.
/** * A drawable resource identifier (in the package's resources) of this * component's icon. From the "icon" attribute or, if not set, 0. */
public int icon;
A drawable resource identifier (in the package's resources) of this component's banner. From the "banner" attribute or, if not set, 0.
/** * A drawable resource identifier (in the package's resources) of this * component's banner. From the "banner" attribute or, if not set, 0. */
public int banner;
A drawable resource identifier (in the package's resources) of this component's logo. Logos may be larger/wider than icons and are displayed by certain UI elements in place of a name or name/icon combination. From the "logo" attribute or, if not set, 0.
/** * A drawable resource identifier (in the package's resources) of this * component's logo. Logos may be larger/wider than icons and are * displayed by certain UI elements in place of a name or name/icon * combination. From the "logo" attribute or, if not set, 0. */
public int ;
Additional meta-data associated with this component. This field will only be filled in if you set the PackageManager.GET_META_DATA flag when requesting the info.
/** * Additional meta-data associated with this component. This field * will only be filled in if you set the * {@link PackageManager#GET_META_DATA} flag when requesting the info. */
public Bundle metaData;
If different of UserHandle.USER_NULL, The icon of this item will be the one of that user.
@hide
/** * If different of UserHandle.USER_NULL, The icon of this item will be the one of that user. * @hide */
public int showUserIcon; public PackageItemInfo() { showUserIcon = UserHandle.USER_NULL; } public PackageItemInfo(PackageItemInfo orig) { name = orig.name; if (name != null) name = name.trim(); packageName = orig.packageName; labelRes = orig.labelRes; nonLocalizedLabel = orig.nonLocalizedLabel; if (nonLocalizedLabel != null) nonLocalizedLabel = nonLocalizedLabel.toString().trim(); icon = orig.icon; banner = orig.banner; logo = orig.logo; metaData = orig.metaData; showUserIcon = orig.showUserIcon; }
Retrieve the current textual label associated with this item. This will call back on the given PackageManager to load the label from the application.
Params:
  • pm – A PackageManager from which the label can be loaded; usually the PackageManager from which you originally retrieved this item.
Returns:Returns a CharSequence containing the item's label. If the item does not have a label, its name is returned.
/** * Retrieve the current textual label associated with this item. This * will call back on the given PackageManager to load the label from * the application. * * @param pm A PackageManager from which the label can be loaded; usually * the PackageManager from which you originally retrieved this item. * * @return Returns a CharSequence containing the item's label. If the * item does not have a label, its name is returned. */
public @NonNull CharSequence loadLabel(@NonNull PackageManager pm) { if (sForceSafeLabels) { return loadSafeLabel(pm); } else { return loadUnsafeLabel(pm); } }
{@hide}
/** {@hide} */
public CharSequence loadUnsafeLabel(PackageManager pm) { if (nonLocalizedLabel != null) { return nonLocalizedLabel; } if (labelRes != 0) { CharSequence label = pm.getText(packageName, labelRes, getApplicationInfo()); if (label != null) { return label.toString().trim(); } } if (name != null) { return name; } return packageName; }
Deprecated use loadSafeLabel(PackageManager, float, int) instead
@hide
/** * Deprecated use loadSafeLabel(PackageManager, float, int) instead * * @hide */
@SystemApi public @NonNull CharSequence loadSafeLabel(@NonNull PackageManager pm) { // loadLabel() always returns non-null String label = loadUnsafeLabel(pm).toString(); // strip HTML tags to avoid <br> and other tags overwriting original message String labelStr = Html.fromHtml(label).toString(); // If the label contains new line characters it may push the UI // down to hide a part of it. Labels shouldn't have new line // characters, so just truncate at the first time one is seen. final int labelLength = Math.min(labelStr.length(), MAX_SAFE_LABEL_LENGTH); final StringBuffer sb = new StringBuffer(labelLength); int offset = 0; while (offset < labelLength) { final int codePoint = labelStr.codePointAt(offset); final int type = Character.getType(codePoint); if (type == Character.LINE_SEPARATOR || type == Character.CONTROL || type == Character.PARAGRAPH_SEPARATOR) { labelStr = labelStr.substring(0, offset); break; } // replace all non-break space to " " in order to be trimmed final int charCount = Character.charCount(codePoint); if (type == Character.SPACE_SEPARATOR) { sb.append(' '); } else { sb.append(labelStr.charAt(offset)); if (charCount == 2) { sb.append(labelStr.charAt(offset + 1)); } } offset += charCount; } labelStr = sb.toString().trim(); if (labelStr.isEmpty()) { return packageName; } TextPaint paint = new TextPaint(); paint.setTextSize(42); return TextUtils.ellipsize(labelStr, paint, MAX_LABEL_SIZE_PX, TextUtils.TruncateAt.END); } private static boolean isNewline(int codePoint) { int type = Character.getType(codePoint); return type == Character.PARAGRAPH_SEPARATOR || type == Character.LINE_SEPARATOR || codePoint == LINE_FEED_CODE_POINT; } private static boolean isWhiteSpace(int codePoint) { return Character.isWhitespace(codePoint) || codePoint == NBSP_CODE_POINT; }
A special string manipulation class. Just records removals and executes the when onString() is called.
/** * A special string manipulation class. Just records removals and executes the when onString() * is called. */
private static class StringWithRemovedChars {
The original string
/** The original string */
private final String mOriginal;
One bit per char in string. If bit is set, character needs to be removed. If whole bit field is not initialized nothing needs to be removed.
/** * One bit per char in string. If bit is set, character needs to be removed. If whole * bit field is not initialized nothing needs to be removed. */
private BitSet mRemovedChars; StringWithRemovedChars(@NonNull String original) { mOriginal = original; }
Mark all chars in a range [firstRemoved - firstNonRemoved[ (not including firstNonRemoved) as removed.
/** * Mark all chars in a range {@code [firstRemoved - firstNonRemoved[} (not including * firstNonRemoved) as removed. */
void removeRange(int firstRemoved, int firstNonRemoved) { if (mRemovedChars == null) { mRemovedChars = new BitSet(mOriginal.length()); } mRemovedChars.set(firstRemoved, firstNonRemoved); }
Remove all characters before firstNonRemoved.
/** * Remove all characters before {@code firstNonRemoved}. */
void removeAllCharBefore(int firstNonRemoved) { if (mRemovedChars == null) { mRemovedChars = new BitSet(mOriginal.length()); } mRemovedChars.set(0, firstNonRemoved); }
Remove all characters after and including firstRemoved.
/** * Remove all characters after and including {@code firstRemoved}. */
void removeAllCharAfter(int firstRemoved) { if (mRemovedChars == null) { mRemovedChars = new BitSet(mOriginal.length()); } mRemovedChars.set(firstRemoved, mOriginal.length()); } @Override public String toString() { // Common case, no chars removed if (mRemovedChars == null) { return mOriginal; } StringBuilder sb = new StringBuilder(mOriginal.length()); for (int i = 0; i < mOriginal.length(); i++) { if (!mRemovedChars.get(i)) { sb.append(mOriginal.charAt(i)); } } return sb.toString(); }
Return length or the original string
/** * Return length or the original string */
int length() { return mOriginal.length(); }
Return if a certain offset of the original string is removed
/** * Return if a certain {@code offset} of the original string is removed */
boolean isRemoved(int offset) { return mRemovedChars != null && mRemovedChars.get(offset); }
Return codePoint of original string at a certain offset
/** * Return codePoint of original string at a certain {@code offset} */
int codePointAt(int offset) { return mOriginal.codePointAt(offset); } }
Load, clean up and truncate label before use.

This method is meant to remove common mistakes and nefarious formatting from strings that are used in sensitive parts of the UI.

This method first treats the string like HTML and then ...

  • Removes new lines or truncates at first new line
  • Trims the white-space off the end
  • Truncates the string to a given length
... if specified.
Params:
  • ellipsizeDip – Assuming maximum length of the string (in dip), assuming font size 42. This is roughly 50 characters for ellipsizeDip == 1000.
    Usually ellipsizing should be left to the view showing the string. If a string is used as an input to another string, it might be useful to control the length of the input string though. 0 disables this feature.
Returns:The safe label
@hide
/** * Load, clean up and truncate label before use. * * <p>This method is meant to remove common mistakes and nefarious formatting from strings that * are used in sensitive parts of the UI. * * <p>This method first treats the string like HTML and then ... * <ul> * <li>Removes new lines or truncates at first new line * <li>Trims the white-space off the end * <li>Truncates the string to a given length * </ul> * ... if specified. * * @param ellipsizeDip Assuming maximum length of the string (in dip), assuming font size 42. * This is roughly 50 characters for {@code ellipsizeDip == 1000}.<br /> * Usually ellipsizing should be left to the view showing the string. If a * string is used as an input to another string, it might be useful to * control the length of the input string though. {@code 0} disables this * feature. * @return The safe label * @hide */
public @NonNull CharSequence loadSafeLabel(@NonNull PackageManager pm, @FloatRange(from = 0) float ellipsizeDip, @SafeLabelFlags int flags) { boolean onlyKeepFirstLine = ((flags & SAFE_LABEL_FLAG_FIRST_LINE) != 0); boolean forceSingleLine = ((flags & SAFE_LABEL_FLAG_SINGLE_LINE) != 0); boolean trim = ((flags & SAFE_LABEL_FLAG_TRIM) != 0); Preconditions.checkNotNull(pm); Preconditions.checkArgument(ellipsizeDip >= 0); Preconditions.checkFlagsArgument(flags, SAFE_LABEL_FLAG_TRIM | SAFE_LABEL_FLAG_SINGLE_LINE | SAFE_LABEL_FLAG_FIRST_LINE); Preconditions.checkArgument(!(onlyKeepFirstLine && forceSingleLine), "Cannot set SAFE_LABEL_FLAG_SINGLE_LINE and SAFE_LABEL_FLAG_FIRST_LINE at the same " + "time"); // loadLabel() always returns non-null String label = loadUnsafeLabel(pm).toString(); // Treat string as HTML. This // - converts HTML symbols: e.g. &szlig; -> ß // - applies some HTML tags: e.g. <br> -> \n // - removes invalid characters such as \b // - removes html styling, such as <b> // - applies html formatting: e.g. a<p>b</p>c -> a\n\nb\n\nc // - replaces some html tags by "object replacement" markers: <img> -> \ufffc // - Removes leading white space // - Removes all trailing white space beside a single space // - Collapses double white space StringWithRemovedChars labelStr = new StringWithRemovedChars( Html.fromHtml(label).toString()); int firstNonWhiteSpace = -1; int firstTrailingWhiteSpace = -1; // Remove new lines (if requested) and control characters. int labelLength = labelStr.length(); for (int offset = 0; offset < labelLength; ) { int codePoint = labelStr.codePointAt(offset); int type = Character.getType(codePoint); int codePointLen = Character.charCount(codePoint); boolean isNewline = isNewline(codePoint); if (offset > MAX_SAFE_LABEL_LENGTH || onlyKeepFirstLine && isNewline) { labelStr.removeAllCharAfter(offset); break; } else if (forceSingleLine && isNewline) { labelStr.removeRange(offset, offset + codePointLen); } else if (type == Character.CONTROL && !isNewline) { labelStr.removeRange(offset, offset + codePointLen); } else if (trim && !isWhiteSpace(codePoint)) { // This is only executed if the code point is not removed if (firstNonWhiteSpace == -1) { firstNonWhiteSpace = offset; } firstTrailingWhiteSpace = offset + codePointLen; } offset += codePointLen; } if (trim) { // Remove leading and trailing white space if (firstNonWhiteSpace == -1) { // No non whitespace found, remove all labelStr.removeAllCharAfter(0); } else { if (firstNonWhiteSpace > 0) { labelStr.removeAllCharBefore(firstNonWhiteSpace); } if (firstTrailingWhiteSpace < labelLength) { labelStr.removeAllCharAfter(firstTrailingWhiteSpace); } } } if (ellipsizeDip == 0) { return labelStr.toString(); } else { // Truncate final TextPaint paint = new TextPaint(); paint.setTextSize(42); return TextUtils.ellipsize(labelStr.toString(), paint, ellipsizeDip, TextUtils.TruncateAt.END); } }
Retrieve the current graphical icon associated with this item. This will call back on the given PackageManager to load the icon from the application.
Params:
  • pm – A PackageManager from which the icon can be loaded; usually the PackageManager from which you originally retrieved this item.
Returns:Returns a Drawable containing the item's icon. If the item does not have an icon, the item's default icon is returned such as the default activity icon.
/** * Retrieve the current graphical icon associated with this item. This * will call back on the given PackageManager to load the icon from * the application. * * @param pm A PackageManager from which the icon can be loaded; usually * the PackageManager from which you originally retrieved this item. * * @return Returns a Drawable containing the item's icon. If the * item does not have an icon, the item's default icon is returned * such as the default activity icon. */
public Drawable loadIcon(PackageManager pm) { return pm.loadItemIcon(this, getApplicationInfo()); }
Retrieve the current graphical icon associated with this item without the addition of a work badge if applicable. This will call back on the given PackageManager to load the icon from the application.
Params:
  • pm – A PackageManager from which the icon can be loaded; usually the PackageManager from which you originally retrieved this item.
Returns:Returns a Drawable containing the item's icon. If the item does not have an icon, the item's default icon is returned such as the default activity icon.
/** * Retrieve the current graphical icon associated with this item without * the addition of a work badge if applicable. * This will call back on the given PackageManager to load the icon from * the application. * * @param pm A PackageManager from which the icon can be loaded; usually * the PackageManager from which you originally retrieved this item. * * @return Returns a Drawable containing the item's icon. If the * item does not have an icon, the item's default icon is returned * such as the default activity icon. */
public Drawable loadUnbadgedIcon(PackageManager pm) { return pm.loadUnbadgedItemIcon(this, getApplicationInfo()); }
Retrieve the current graphical banner associated with this item. This will call back on the given PackageManager to load the banner from the application.
Params:
  • pm – A PackageManager from which the banner can be loaded; usually the PackageManager from which you originally retrieved this item.
Returns:Returns a Drawable containing the item's banner. If the item does not have a banner, this method will return null.
/** * Retrieve the current graphical banner associated with this item. This * will call back on the given PackageManager to load the banner from * the application. * * @param pm A PackageManager from which the banner can be loaded; usually * the PackageManager from which you originally retrieved this item. * * @return Returns a Drawable containing the item's banner. If the item * does not have a banner, this method will return null. */
public Drawable loadBanner(PackageManager pm) { if (banner != 0) { Drawable dr = pm.getDrawable(packageName, banner, getApplicationInfo()); if (dr != null) { return dr; } } return loadDefaultBanner(pm); }
Retrieve the default graphical icon associated with this item.
Params:
  • pm – A PackageManager from which the icon can be loaded; usually the PackageManager from which you originally retrieved this item.
Returns:Returns a Drawable containing the item's default icon such as the default activity icon.
@hide
/** * Retrieve the default graphical icon associated with this item. * * @param pm A PackageManager from which the icon can be loaded; usually * the PackageManager from which you originally retrieved this item. * * @return Returns a Drawable containing the item's default icon * such as the default activity icon. * * @hide */
public Drawable loadDefaultIcon(PackageManager pm) { return pm.getDefaultActivityIcon(); }
Retrieve the default graphical banner associated with this item.
Params:
  • pm – A PackageManager from which the banner can be loaded; usually the PackageManager from which you originally retrieved this item.
Returns:Returns a Drawable containing the item's default banner or null if no default logo is available.
@hide
/** * Retrieve the default graphical banner associated with this item. * * @param pm A PackageManager from which the banner can be loaded; usually * the PackageManager from which you originally retrieved this item. * * @return Returns a Drawable containing the item's default banner * or null if no default logo is available. * * @hide */
protected Drawable loadDefaultBanner(PackageManager pm) { return null; }
Retrieve the current graphical logo associated with this item. This will call back on the given PackageManager to load the logo from the application.
Params:
  • pm – A PackageManager from which the logo can be loaded; usually the PackageManager from which you originally retrieved this item.
Returns:Returns a Drawable containing the item's logo. If the item does not have a logo, this method will return null.
/** * Retrieve the current graphical logo associated with this item. This * will call back on the given PackageManager to load the logo from * the application. * * @param pm A PackageManager from which the logo can be loaded; usually * the PackageManager from which you originally retrieved this item. * * @return Returns a Drawable containing the item's logo. If the item * does not have a logo, this method will return null. */
public Drawable loadLogo(PackageManager pm) { if (logo != 0) { Drawable d = pm.getDrawable(packageName, logo, getApplicationInfo()); if (d != null) { return d; } } return loadDefaultLogo(pm); }
Retrieve the default graphical logo associated with this item.
Params:
  • pm – A PackageManager from which the logo can be loaded; usually the PackageManager from which you originally retrieved this item.
Returns:Returns a Drawable containing the item's default logo or null if no default logo is available.
@hide
/** * Retrieve the default graphical logo associated with this item. * * @param pm A PackageManager from which the logo can be loaded; usually * the PackageManager from which you originally retrieved this item. * * @return Returns a Drawable containing the item's default logo * or null if no default logo is available. * * @hide */
protected Drawable loadDefaultLogo(PackageManager pm) { return null; }
Load an XML resource attached to the meta-data of this item. This will retrieved the name meta-data entry, and if defined call back on the given PackageManager to load its XML file from the application.
Params:
  • pm – A PackageManager from which the XML can be loaded; usually the PackageManager from which you originally retrieved this item.
  • name – Name of the meta-date you would like to load.
Returns:Returns an XmlPullParser you can use to parse the XML file assigned as the given meta-data. If the meta-data name is not defined or the XML resource could not be found, null is returned.
/** * Load an XML resource attached to the meta-data of this item. This will * retrieved the name meta-data entry, and if defined call back on the * given PackageManager to load its XML file from the application. * * @param pm A PackageManager from which the XML can be loaded; usually * the PackageManager from which you originally retrieved this item. * @param name Name of the meta-date you would like to load. * * @return Returns an XmlPullParser you can use to parse the XML file * assigned as the given meta-data. If the meta-data name is not defined * or the XML resource could not be found, null is returned. */
public XmlResourceParser loadXmlMetaData(PackageManager pm, String name) { if (metaData != null) { int resid = metaData.getInt(name); if (resid != 0) { return pm.getXml(packageName, resid, getApplicationInfo()); } } return null; }
@hideFlag for dumping: include all details.
/** * @hide Flag for dumping: include all details. */
public static final int DUMP_FLAG_DETAILS = 1<<0;
@hideFlag for dumping: include nested ApplicationInfo.
/** * @hide Flag for dumping: include nested ApplicationInfo. */
public static final int DUMP_FLAG_APPLICATION = 1<<1;
@hideFlag for dumping: all flags to dump everything.
/** * @hide Flag for dumping: all flags to dump everything. */
public static final int DUMP_FLAG_ALL = DUMP_FLAG_DETAILS | DUMP_FLAG_APPLICATION; protected void dumpFront(Printer pw, String prefix) { if (name != null) { pw.println(prefix + "name=" + name); } pw.println(prefix + "packageName=" + packageName); if (labelRes != 0 || nonLocalizedLabel != null || icon != 0 || banner != 0) { pw.println(prefix + "labelRes=0x" + Integer.toHexString(labelRes) + " nonLocalizedLabel=" + nonLocalizedLabel + " icon=0x" + Integer.toHexString(icon) + " banner=0x" + Integer.toHexString(banner)); } } protected void dumpBack(Printer pw, String prefix) { // no back here } public void writeToParcel(Parcel dest, int parcelableFlags) { dest.writeString(name); dest.writeString(packageName); dest.writeInt(labelRes); TextUtils.writeToParcel(nonLocalizedLabel, dest, parcelableFlags); dest.writeInt(icon); dest.writeInt(logo); dest.writeBundle(metaData); dest.writeInt(banner); dest.writeInt(showUserIcon); }
@hide
/** * @hide */
public void writeToProto(ProtoOutputStream proto, long fieldId) { long token = proto.start(fieldId); if (name != null) { proto.write(PackageItemInfoProto.NAME, name); } proto.write(PackageItemInfoProto.PACKAGE_NAME, packageName); if (labelRes != 0 || nonLocalizedLabel != null || icon != 0 || banner != 0) { proto.write(PackageItemInfoProto.LABEL_RES, labelRes); proto.write(PackageItemInfoProto.NON_LOCALIZED_LABEL, nonLocalizedLabel.toString()); proto.write(PackageItemInfoProto.ICON, icon); proto.write(PackageItemInfoProto.BANNER, banner); } proto.end(token); } protected PackageItemInfo(Parcel source) { name = source.readString(); packageName = source.readString(); labelRes = source.readInt(); nonLocalizedLabel = TextUtils.CHAR_SEQUENCE_CREATOR.createFromParcel(source); icon = source.readInt(); logo = source.readInt(); metaData = source.readBundle(); banner = source.readInt(); showUserIcon = source.readInt(); }
Get the ApplicationInfo for the application to which this item belongs, if available, otherwise returns null.
Returns:Returns the ApplicationInfo of this item, or null if not known.
@hide
/** * Get the ApplicationInfo for the application to which this item belongs, * if available, otherwise returns null. * * @return Returns the ApplicationInfo of this item, or null if not known. * * @hide */
protected ApplicationInfo getApplicationInfo() { return null; } public static class DisplayNameComparator implements Comparator<PackageItemInfo> { public DisplayNameComparator(PackageManager pm) { mPM = pm; } public final int compare(PackageItemInfo aa, PackageItemInfo ab) { CharSequence sa = aa.loadLabel(mPM); if (sa == null) sa = aa.name; CharSequence sb = ab.loadLabel(mPM); if (sb == null) sb = ab.name; return sCollator.compare(sa.toString(), sb.toString()); } private final Collator sCollator = Collator.getInstance(); private PackageManager mPM; } }