/* * Copyright (C) 2016 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.launcher3.model; import static com.android.launcher3.LauncherSettings.Favorites.ITEM_TYPE_DEEP_SHORTCUT; import static com.android.launcher3.model.BgDataModel.Callbacks.FLAG_PRIVATE_PROFILE_QUIET_MODE_ENABLED; import static com.android.launcher3.model.BgDataModel.Callbacks.FLAG_QUIET_MODE_ENABLED; import static com.android.launcher3.model.BgDataModel.Callbacks.FLAG_WORK_PROFILE_QUIET_MODE_ENABLED; import static com.android.launcher3.model.ModelUtils.WIDGET_FILTER; import static com.android.launcher3.model.data.ItemInfoWithIcon.FLAG_ARCHIVED; import static com.android.launcher3.model.data.LauncherAppWidgetInfo.FLAG_PROVIDER_NOT_READY; import static com.android.launcher3.model.data.WorkspaceItemInfo.FLAG_AUTOINSTALL_ICON; import static com.android.launcher3.model.data.WorkspaceItemInfo.FLAG_RESTORED_ICON; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.pm.LauncherActivityInfo; import android.content.pm.LauncherApps; import android.content.pm.ShortcutInfo; import android.os.UserHandle; import android.os.UserManager; import android.util.Log; import androidx.annotation.NonNull; import com.android.launcher3.Flags; import com.android.launcher3.LauncherModel.ModelUpdateTask; import com.android.launcher3.LauncherSettings.Favorites; import com.android.launcher3.config.FeatureFlags; import com.android.launcher3.icons.IconCache; import com.android.launcher3.logging.FileLog; import com.android.launcher3.model.data.ItemInfo; import com.android.launcher3.model.data.LauncherAppWidgetInfo; import com.android.launcher3.model.data.WorkspaceItemInfo; import com.android.launcher3.pm.PackageInstallInfo; import com.android.launcher3.pm.UserCache; import com.android.launcher3.shortcuts.ShortcutRequest; import com.android.launcher3.util.ApiWrapper; import com.android.launcher3.util.FlagOp; import com.android.launcher3.util.IntSet; import com.android.launcher3.util.ItemInfoMatcher; import com.android.launcher3.util.PackageManagerHelper; import com.android.launcher3.util.PackageUserKey; import com.android.launcher3.util.SafeCloseable; import java.util.ArrayList; import java.util.Arrays; import java.util.Collections; import java.util.HashMap; import java.util.HashSet; import java.util.List; import java.util.Objects; import java.util.function.Predicate; import java.util.stream.Collectors; /** * Handles updates due to changes in package manager (app installed/updated/removed) * or when a user availability changes. */ @SuppressWarnings("NewApi") public class PackageUpdatedTask implements ModelUpdateTask { // TODO(b/290090023): Set to false after root causing is done. private static final String TAG = "PackageUpdatedTask"; private static final boolean DEBUG = true; public static final int OP_NONE = 0; public static final int OP_ADD = 1; public static final int OP_UPDATE = 2; public static final int OP_REMOVE = 3; // uninstalled public static final int OP_UNAVAILABLE = 4; // external media unmounted public static final int OP_SUSPEND = 5; // package suspended public static final int OP_UNSUSPEND = 6; // package unsuspended public static final int OP_USER_AVAILABILITY_CHANGE = 7; // user available/unavailable private final int mOp; @NonNull private final UserHandle mUser; @NonNull private final String[] mPackages; public PackageUpdatedTask(final int op, @NonNull final UserHandle user, @NonNull final String... packages) { mOp = op; mUser = user; mPackages = packages; } @Override public void execute(@NonNull ModelTaskController taskController, @NonNull BgDataModel dataModel, @NonNull AllAppsList appsList) { final Context context = taskController.getContext(); final IconCache iconCache = taskController.getIconCache(); final String[] packages = mPackages; final int packageCount = packages.length; final FlagOp flagOp; final HashSet packageSet = new HashSet<>(Arrays.asList(packages)); final Predicate matcher = mOp == OP_USER_AVAILABILITY_CHANGE ? ItemInfoMatcher.ofUser(mUser) // We want to update all packages for this user : ItemInfoMatcher.ofPackages(packageSet, mUser); final HashSet removedComponents = new HashSet<>(); final HashMap> activitiesLists = new HashMap<>(); if (DEBUG) { Log.d(TAG, "Package updated: mOp=" + getOpString() + " packages=" + Arrays.toString(packages) + ", user=" + mUser); } switch (mOp) { case OP_ADD: { for (int i = 0; i < packageCount; i++) { iconCache.updateIconsForPkg(packages[i], mUser); if (FeatureFlags.PROMISE_APPS_IN_ALL_APPS.get()) { if (DEBUG) { Log.d(TAG, "OP_ADD: PROMISE_APPS_IN_ALL_APPS enabled:" + " removing promise icon apps from package=" + packages[i]); } appsList.removePackage(packages[i], mUser); } activitiesLists.put(packages[i], appsList.addPackage(context, packages[i], mUser)); } flagOp = FlagOp.NO_OP.removeFlag(WorkspaceItemInfo.FLAG_DISABLED_NOT_AVAILABLE); break; } case OP_UPDATE: try (SafeCloseable t = appsList.trackRemoves(a -> { Log.d(TAG, "OP_UPDATE - AllAppsList.trackRemoves callback:" + " removed component=" + a.componentName + " id=" + a.id + " Look for earlier AllAppsList logs to find more information."); removedComponents.add(a.componentName); })) { for (int i = 0; i < packageCount; i++) { iconCache.updateIconsForPkg(packages[i], mUser); activitiesLists.put(packages[i], appsList.updatePackage(context, packages[i], mUser)); } } // Since package was just updated, the target must be available now. flagOp = FlagOp.NO_OP.removeFlag(WorkspaceItemInfo.FLAG_DISABLED_NOT_AVAILABLE); break; case OP_REMOVE: { for (int i = 0; i < packageCount; i++) { iconCache.removeIconsForPkg(packages[i], mUser); } // Fall through } case OP_UNAVAILABLE: for (int i = 0; i < packageCount; i++) { if (DEBUG) { Log.d(TAG, getOpString() + ": removing package=" + packages[i]); } appsList.removePackage(packages[i], mUser); } flagOp = FlagOp.NO_OP.addFlag(WorkspaceItemInfo.FLAG_DISABLED_NOT_AVAILABLE); break; case OP_SUSPEND: case OP_UNSUSPEND: flagOp = FlagOp.NO_OP.setFlag( WorkspaceItemInfo.FLAG_DISABLED_SUSPENDED, mOp == OP_SUSPEND); appsList.updateDisabledFlags(matcher, flagOp); break; case OP_USER_AVAILABILITY_CHANGE: { UserManagerState ums = new UserManagerState(); UserManager userManager = context.getSystemService(UserManager.class); ums.init(UserCache.INSTANCE.get(context), userManager); boolean isUserQuiet = ums.isUserQuiet(mUser); flagOp = FlagOp.NO_OP.setFlag( WorkspaceItemInfo.FLAG_DISABLED_QUIET_USER, isUserQuiet); appsList.updateDisabledFlags(matcher, flagOp); if (Flags.enablePrivateSpace()) { UserCache userCache = UserCache.INSTANCE.get(context); if (userCache.getUserInfo(mUser).isWork()) { appsList.setFlags(FLAG_WORK_PROFILE_QUIET_MODE_ENABLED, isUserQuiet); } else if (userCache.getUserInfo(mUser).isPrivate()) { appsList.setFlags(FLAG_PRIVATE_PROFILE_QUIET_MODE_ENABLED, isUserQuiet); } } else { // We are not synchronizing here, as int operations are atomic appsList.setFlags(FLAG_QUIET_MODE_ENABLED, ums.isAnyProfileQuietModeEnabled()); } break; } default: flagOp = FlagOp.NO_OP; break; } taskController.bindApplicationsIfNeeded(); final IntSet removedShortcuts = new IntSet(); // Shortcuts to keep even if the corresponding app was removed final IntSet forceKeepShortcuts = new IntSet(); // Update shortcut infos if (mOp == OP_ADD || flagOp != FlagOp.NO_OP) { final ArrayList updatedWorkspaceItems = new ArrayList<>(); // For system apps, package manager send OP_UPDATE when an app is enabled. final boolean isNewApkAvailable = mOp == OP_ADD || mOp == OP_UPDATE; synchronized (dataModel) { dataModel.forAllWorkspaceItemInfos(mUser, itemInfo -> { boolean infoUpdated = false; boolean shortcutUpdated = false; ComponentName cn = itemInfo.getTargetComponent(); if (cn != null && matcher.test(itemInfo)) { String packageName = cn.getPackageName(); if (itemInfo.hasStatusFlag(WorkspaceItemInfo.FLAG_SUPPORTS_WEB_UI)) { forceKeepShortcuts.add(itemInfo.id); if (mOp == OP_REMOVE) { return; } } if (itemInfo.isPromise() && isNewApkAvailable) { boolean isTargetValid = !cn.getClassName().equals( IconCache.EMPTY_CLASS_NAME); if (itemInfo.itemType == ITEM_TYPE_DEEP_SHORTCUT) { int requestQuery = ShortcutRequest.PINNED; if (Flags.restoreArchivedShortcuts()) { // Avoid race condition where shortcut service has no record of // unarchived shortcut being pinned after restore. // Launcher should be source-of-truth for if shortcut is pinned. requestQuery = ShortcutRequest.ALL; } List shortcut = new ShortcutRequest(context, mUser) .forPackage(cn.getPackageName(), itemInfo.getDeepShortcutId()) .query(requestQuery); if (shortcut.isEmpty()) { isTargetValid = false; if (DEBUG) { Log.d(TAG, "Shortcut not found for updated" + " package=" + itemInfo.getTargetPackage() + ", isArchived=" + itemInfo.isArchived()); } } else { if (DEBUG) { Log.d(TAG, "Found shortcut for updated" + " package=" + itemInfo.getTargetPackage() + ", isTargetValid=" + isTargetValid + ", isArchived=" + itemInfo.isArchived()); } itemInfo.updateFromDeepShortcutInfo(shortcut.get(0), context); infoUpdated = true; } } else if (isTargetValid) { isTargetValid = context.getSystemService(LauncherApps.class) .isActivityEnabled(cn, mUser); } if (!isTargetValid && (itemInfo.hasStatusFlag( FLAG_RESTORED_ICON | FLAG_AUTOINSTALL_ICON) || itemInfo.isArchived())) { if (updateWorkspaceItemIntent(context, itemInfo, packageName)) { infoUpdated = true; } else if (shouldRemoveRestoredShortcut(itemInfo)) { removedShortcuts.add(itemInfo.id); if (DEBUG) { FileLog.w(TAG, "Removing restored shortcut promise icon" + " that no longer points to valid component." + " id=" + itemInfo.id + ", package=" + itemInfo.getTargetPackage() + ", status=" + itemInfo.status + ", isArchived=" + itemInfo.isArchived()); } return; } } else if (!isTargetValid) { removedShortcuts.add(itemInfo.id); if (DEBUG) { FileLog.w(TAG, "Removing shortcut that no longer points to" + " valid component." + " id=" + itemInfo.id + " package=" + itemInfo.getTargetPackage() + " status=" + itemInfo.status); } return; } else { itemInfo.status = WorkspaceItemInfo.DEFAULT; infoUpdated = true; } } else if (isNewApkAvailable && removedComponents.contains(cn)) { if (updateWorkspaceItemIntent(context, itemInfo, packageName)) { infoUpdated = true; } } if (isNewApkAvailable) { List activities = activitiesLists.get( packageName); // TODO: See if we can migrate this to // AppInfo#updateRuntimeFlagsForActivityTarget itemInfo.setProgressLevel( activities == null || activities.isEmpty() ? 100 : PackageManagerHelper.getLoadingProgress( activities.get(0)), PackageInstallInfo.STATUS_INSTALLED_DOWNLOADING); // In case an app is archived, we need to make sure that archived state // in WorkspaceItemInfo is refreshed. if (Flags.enableSupportForArchiving() && !activities.isEmpty()) { boolean newArchivalState = activities.get(0) .getActivityInfo().isArchived; if (newArchivalState != itemInfo.isArchived()) { itemInfo.runtimeStatusFlags ^= FLAG_ARCHIVED; infoUpdated = true; } } if (itemInfo.itemType == Favorites.ITEM_TYPE_APPLICATION) { if (activities != null && !activities.isEmpty()) { itemInfo.setNonResizeable(ApiWrapper.INSTANCE.get(context) .isNonResizeableActivity(activities.get(0))); } iconCache.getTitleAndIcon( itemInfo, itemInfo.getMatchingLookupFlag()); infoUpdated = true; } } int oldRuntimeFlags = itemInfo.runtimeStatusFlags; itemInfo.runtimeStatusFlags = flagOp.apply(itemInfo.runtimeStatusFlags); if (itemInfo.runtimeStatusFlags != oldRuntimeFlags) { shortcutUpdated = true; } } if (infoUpdated || shortcutUpdated) { updatedWorkspaceItems.add(itemInfo); } if (infoUpdated && itemInfo.id != ItemInfo.NO_ID) { taskController.getModelWriter().updateItemInDatabase(itemInfo); } }); dataModel.itemsIdMap.stream() .filter(WIDGET_FILTER) .filter(item -> mUser.equals(item.user)) .map(item -> (LauncherAppWidgetInfo) item) .filter(widget -> widget.hasRestoreFlag(FLAG_PROVIDER_NOT_READY) && packageSet.contains(widget.providerName.getPackageName())) .forEach(widgetInfo -> { widgetInfo.restoreStatus &= ~FLAG_PROVIDER_NOT_READY & ~LauncherAppWidgetInfo.FLAG_RESTORE_STARTED; // adding this flag ensures that launcher shows 'click to setup' // if the widget has a config activity. In case there is no config // activity, it will be marked as 'restored' during bind. widgetInfo.restoreStatus |= LauncherAppWidgetInfo.FLAG_UI_NOT_READY; widgetInfo.installProgress = 100; updatedWorkspaceItems.add(widgetInfo); taskController.getModelWriter().updateItemInDatabase(widgetInfo); }); } taskController.bindUpdatedWorkspaceItems(updatedWorkspaceItems); if (!removedShortcuts.isEmpty()) { taskController.deleteAndBindComponentsRemoved( ItemInfoMatcher.ofItemIds(removedShortcuts), "removing shortcuts with invalid target components." + " ids=" + removedShortcuts); } } final HashSet removedPackages = new HashSet<>(); if (mOp == OP_REMOVE) { // Mark all packages in the broadcast to be removed Collections.addAll(removedPackages, packages); if (DEBUG) { Log.d(TAG, "OP_REMOVE: removing packages=" + Arrays.toString(packages)); } // No need to update the removedComponents as // removedPackages is a super-set of removedComponents } else if (mOp == OP_UPDATE) { // Mark disabled packages in the broadcast to be removed final LauncherApps launcherApps = context.getSystemService(LauncherApps.class); for (int i = 0; i < packageCount; i++) { if (!launcherApps.isPackageEnabled(packages[i], mUser)) { if (DEBUG) { Log.d(TAG, "OP_UPDATE:" + " package " + packages[i] + " is disabled, removing package."); } removedPackages.add(packages[i]); } } } if (!removedPackages.isEmpty() || !removedComponents.isEmpty()) { Predicate removeMatch = ItemInfoMatcher.ofPackages(removedPackages, mUser) .or(ItemInfoMatcher.ofComponents(removedComponents, mUser)) .and(ItemInfoMatcher.ofItemIds(forceKeepShortcuts).negate()); taskController.deleteAndBindComponentsRemoved(removeMatch, "removed because the corresponding package or component is removed. " + "mOp=" + mOp + " removedPackages=" + removedPackages.stream().collect( Collectors.joining(",", "[", "]")) + " removedComponents=" + removedComponents.stream() .filter(Objects::nonNull).map(ComponentName::toShortString) .collect(Collectors.joining(",", "[", "]"))); // Remove any queued items from the install queue ItemInstallQueue.INSTANCE.get(context) .removeFromInstallQueue(removedPackages, mUser); } if (mOp == OP_ADD) { // Load widgets for the new package. Changes due to app updates are handled through // AppWidgetHost events, this is just to initialize the long-press options. for (int i = 0; i < packageCount; i++) { dataModel.widgetsModel.update(new PackageUserKey(packages[i], mUser)); } taskController.bindUpdatedWidgets(dataModel); } } /** * Updates {@param si}'s intent to point to a new ComponentName. * @return Whether the shortcut intent was changed. */ private boolean updateWorkspaceItemIntent(Context context, WorkspaceItemInfo si, String packageName) { if (si.itemType == ITEM_TYPE_DEEP_SHORTCUT) { // Do not update intent for deep shortcuts as they contain additional information // about the shortcut. return false; } // Try to find the best match activity. Intent intent = PackageManagerHelper.INSTANCE.get(context) .getAppLaunchIntent(packageName, mUser); if (intent != null) { si.intent = intent; si.status = WorkspaceItemInfo.DEFAULT; return true; } return false; } private boolean shouldRemoveRestoredShortcut(WorkspaceItemInfo itemInfo) { if (itemInfo.hasPromiseIconUi() && !Flags.restoreArchivedShortcuts()) { return true; } return Flags.restoreArchivedShortcuts() && !itemInfo.isArchived() && itemInfo.itemType == ITEM_TYPE_DEEP_SHORTCUT; } private String getOpString() { return switch (mOp) { case OP_NONE -> "NONE"; case OP_ADD -> "ADD"; case OP_UPDATE -> "UPDATE"; case OP_REMOVE -> "REMOVE"; case OP_UNAVAILABLE -> "UNAVAILABLE"; case OP_SUSPEND -> "SUSPEND"; case OP_UNSUSPEND -> "UNSUSPEND"; case OP_USER_AVAILABILITY_CHANGE -> "USER_AVAILABILITY_CHANGE"; default -> "UNKNOWN"; }; } }