| /* |
| * Copyright (C) 2015 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 com.android.settingslib.applications; |
| |
| import android.app.Application; |
| import android.content.ComponentName; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.content.IntentFilter; |
| import android.content.pm.ApplicationInfo; |
| import android.content.pm.PackageInfo; |
| import android.content.pm.PackageManager; |
| import android.content.pm.ResolveInfo; |
| import android.graphics.drawable.Drawable; |
| import android.hardware.usb.IUsbManager; |
| import android.net.Uri; |
| import android.os.Environment; |
| import android.os.RemoteException; |
| import android.os.SystemProperties; |
| import android.os.UserHandle; |
| import android.os.UserManager; |
| import android.text.TextUtils; |
| import android.util.Log; |
| |
| import com.android.settingslib.R; |
| import com.android.settingslib.Utils; |
| import com.android.settingslib.applications.instantapps.InstantAppDataProvider; |
| import com.android.settingslib.utils.ThreadUtils; |
| |
| import java.util.ArrayList; |
| import java.util.List; |
| |
| public class AppUtils { |
| private static final String TAG = "AppUtils"; |
| |
| /** |
| * This should normally only be set in robolectric tests, to avoid getting a method not found |
| * exception when calling the isInstantApp method of the ApplicationInfo class, because |
| * robolectric does not yet have an implementation of it. |
| */ |
| private static InstantAppDataProvider sInstantAppDataProvider = null; |
| |
| private static final Intent sBrowserIntent; |
| |
| static { |
| sBrowserIntent = new Intent() |
| .setAction(Intent.ACTION_VIEW) |
| .addCategory(Intent.CATEGORY_BROWSABLE) |
| .setData(Uri.parse("http:")); |
| } |
| |
| public static CharSequence getLaunchByDefaultSummary(ApplicationsState.AppEntry appEntry, |
| IUsbManager usbManager, PackageManager pm, Context context) { |
| String packageName = appEntry.info.packageName; |
| boolean hasPreferred = hasPreferredActivities(pm, packageName) |
| || hasUsbDefaults(usbManager, packageName); |
| int status = pm.getIntentVerificationStatusAsUser(packageName, UserHandle.myUserId()); |
| // consider a visible current link-handling state to be any explicitly designated behavior |
| boolean hasDomainURLsPreference = |
| status != PackageManager.INTENT_FILTER_DOMAIN_VERIFICATION_STATUS_UNDEFINED; |
| return context.getString(hasPreferred || hasDomainURLsPreference |
| ? R.string.launch_defaults_some |
| : R.string.launch_defaults_none); |
| } |
| |
| public static boolean hasUsbDefaults(IUsbManager usbManager, String packageName) { |
| try { |
| if (usbManager != null) { |
| return usbManager.hasDefaults(packageName, UserHandle.myUserId()); |
| } |
| } catch (RemoteException e) { |
| Log.e(TAG, "mUsbManager.hasDefaults", e); |
| } |
| return false; |
| } |
| |
| public static boolean hasPreferredActivities(PackageManager pm, String packageName) { |
| // Get list of preferred activities |
| List<ComponentName> prefActList = new ArrayList<>(); |
| // Intent list cannot be null. so pass empty list |
| List<IntentFilter> intentList = new ArrayList<>(); |
| pm.getPreferredActivities(intentList, prefActList, packageName); |
| Log.d(TAG, "Have " + prefActList.size() + " number of activities in preferred list"); |
| return prefActList.size() > 0; |
| } |
| |
| /** |
| * Returns a boolean indicating whether the given package should be considered an instant app |
| */ |
| public static boolean isInstant(ApplicationInfo info) { |
| if (sInstantAppDataProvider != null) { |
| if (sInstantAppDataProvider.isInstantApp(info)) { |
| return true; |
| } |
| } else if (info.isInstantApp()) { |
| return true; |
| } |
| |
| // For debugging/testing, we support setting the following property to a comma-separated |
| // list of search terms (typically, but not necessarily, full package names) to match |
| // against the package names of the app. |
| String propVal = SystemProperties.get("settingsdebug.instant.packages"); |
| if (propVal != null && !propVal.isEmpty() && info.packageName != null) { |
| String[] searchTerms = propVal.split(","); |
| if (searchTerms != null) { |
| for (String term : searchTerms) { |
| if (info.packageName.contains(term)) { |
| return true; |
| } |
| } |
| } |
| } |
| return false; |
| } |
| |
| /** Returns the label for a given package. */ |
| public static CharSequence getApplicationLabel( |
| PackageManager packageManager, String packageName) { |
| return com.android.settingslib.utils.applications.AppUtils |
| .getApplicationLabel(packageManager, packageName); |
| } |
| |
| /** |
| * Returns a boolean indicating whether the given package is a hidden system module |
| */ |
| public static boolean isHiddenSystemModule(Context context, String packageName) { |
| return ApplicationsState.getInstance((Application) context.getApplicationContext()) |
| .isHiddenModule(packageName); |
| } |
| |
| /** |
| * Returns a boolean indicating whether a given package is a system module. |
| */ |
| public static boolean isSystemModule(Context context, String packageName) { |
| return ApplicationsState.getInstance((Application) context.getApplicationContext()) |
| .isSystemModule(packageName); |
| } |
| |
| /** |
| * Returns a boolean indicating whether a given package is a mainline module. |
| */ |
| public static boolean isMainlineModule(PackageManager pm, String packageName) { |
| // Check if the package is listed among the system modules. |
| try { |
| pm.getModuleInfo(packageName, 0 /* flags */); |
| return true; |
| } catch (PackageManager.NameNotFoundException e) { |
| //pass |
| } |
| |
| try { |
| final PackageInfo pkg = pm.getPackageInfo(packageName, 0 /* flags */); |
| // Check if the package is contained in an APEX. There is no public API to properly |
| // check whether a given APK package comes from an APEX registered as module. |
| // Therefore we conservatively assume that any package scanned from an /apex path is |
| // a system package. |
| return pkg.applicationInfo.sourceDir.startsWith( |
| Environment.getApexDirectory().getAbsolutePath()); |
| } catch (PackageManager.NameNotFoundException e) { |
| return false; |
| } |
| } |
| |
| /** |
| * Returns a content description of an app name which distinguishes a personal app from a |
| * work app for accessibility purpose. |
| * If the app is in a work profile, then add a "work" prefix to the app name. |
| */ |
| public static String getAppContentDescription(Context context, String packageName, |
| int userId) { |
| return com.android.settingslib.utils.applications.AppUtils.getAppContentDescription(context, |
| packageName, userId); |
| } |
| |
| /** |
| * Returns a boolean indicating whether a given package is a browser app. |
| * |
| * An app is a "browser" if it has an activity resolution that wound up |
| * marked with the 'handleAllWebDataURI' flag. |
| */ |
| public static boolean isBrowserApp(Context context, String packageName, int userId) { |
| sBrowserIntent.setPackage(packageName); |
| final List<ResolveInfo> list = context.getPackageManager().queryIntentActivitiesAsUser( |
| sBrowserIntent, PackageManager.MATCH_ALL, userId); |
| for (ResolveInfo info : list) { |
| if (info.activityInfo != null && info.handleAllWebDataURI) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| /** |
| * Returns a boolean indicating whether a given package is a default browser. |
| * |
| * @param packageName a given package. |
| * @return true if the given package is default browser. |
| */ |
| public static boolean isDefaultBrowser(Context context, String packageName) { |
| final String defaultBrowserPackage = |
| context.getPackageManager().getDefaultBrowserPackageNameAsUser( |
| UserHandle.myUserId()); |
| return TextUtils.equals(packageName, defaultBrowserPackage); |
| } |
| |
| /** |
| * Get the app icon by app entry. |
| * |
| * @param context caller's context |
| * @param appEntry AppEntry of ApplicationsState |
| * @return app icon of the app entry |
| */ |
| public static Drawable getIcon(Context context, ApplicationsState.AppEntry appEntry) { |
| if (appEntry == null || appEntry.info == null) { |
| return null; |
| } |
| |
| final AppIconCacheManager appIconCacheManager = AppIconCacheManager.getInstance(); |
| final String packageName = appEntry.info.packageName; |
| final int uid = appEntry.info.uid; |
| |
| Drawable icon = appIconCacheManager.get(packageName, uid); |
| if (icon == null) { |
| if (appEntry.apkFile != null && appEntry.apkFile.exists()) { |
| icon = Utils.getBadgedIcon(context, appEntry.info); |
| appIconCacheManager.put(packageName, uid, icon); |
| } else { |
| setAppEntryMounted(appEntry, /* mounted= */ false); |
| icon = context.getDrawable( |
| com.android.internal.R.drawable.sym_app_on_sd_unavailable_icon); |
| } |
| } else if (!appEntry.mounted && appEntry.apkFile != null && appEntry.apkFile.exists()) { |
| // If the app wasn't mounted but is now mounted, reload its icon. |
| setAppEntryMounted(appEntry, /* mounted= */ true); |
| icon = Utils.getBadgedIcon(context, appEntry.info); |
| appIconCacheManager.put(packageName, uid, icon); |
| } |
| |
| return icon; |
| } |
| |
| /** |
| * Get the app icon from cache by app entry. |
| * |
| * @param appEntry AppEntry of ApplicationsState |
| * @return app icon of the app entry |
| */ |
| public static Drawable getIconFromCache(ApplicationsState.AppEntry appEntry) { |
| return appEntry == null || appEntry.info == null ? null |
| : AppIconCacheManager.getInstance().get( |
| appEntry.info.packageName, |
| appEntry.info.uid); |
| } |
| |
| /** |
| * Preload the top N icons of app entry list. |
| * |
| * @param context caller's context |
| * @param appEntries AppEntry list of ApplicationsState |
| * @param number the number of Top N icons of the appEntries |
| */ |
| public static void preloadTopIcons(Context context, |
| ArrayList<ApplicationsState.AppEntry> appEntries, int number) { |
| if (appEntries == null || appEntries.isEmpty() || number <= 0) { |
| return; |
| } |
| |
| for (int i = 0; i < Math.min(appEntries.size(), number); i++) { |
| final ApplicationsState.AppEntry entry = appEntries.get(i); |
| ThreadUtils.postOnBackgroundThread(() -> { |
| getIcon(context, entry); |
| }); |
| } |
| } |
| |
| /** |
| * Returns a boolean indicating whether this app is installed or not. |
| * |
| * @param appEntry AppEntry of ApplicationsState. |
| * @return true if the app is in installed state. |
| */ |
| public static boolean isAppInstalled(ApplicationsState.AppEntry appEntry) { |
| if (appEntry == null || appEntry.info == null) { |
| return false; |
| } |
| return (appEntry.info.flags & ApplicationInfo.FLAG_INSTALLED) != 0; |
| } |
| |
| private static void setAppEntryMounted(ApplicationsState.AppEntry appEntry, boolean mounted) { |
| if (appEntry.mounted != mounted) { |
| synchronized (appEntry) { |
| appEntry.mounted = mounted; |
| } |
| } |
| } |
| |
| /** |
| * Returns clone user profile id if present. Returns -1 if not present. |
| */ |
| public static int getCloneUserId(Context context) { |
| UserManager userManager = context.getSystemService(UserManager.class); |
| for (UserHandle userHandle : userManager.getUserProfiles()) { |
| if (userManager.getUserInfo(userHandle.getIdentifier()).isCloneProfile()) { |
| return userHandle.getIdentifier(); |
| } |
| } |
| return -1; |
| } |
| } |