/* * Copyright (C) 2021 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.CONTAINER_WIDGETS_PREDICTION; import android.app.prediction.AppTarget; import android.text.TextUtils; import androidx.annotation.NonNull; import com.android.launcher3.LauncherAppState; import com.android.launcher3.model.BgDataModel.FixedContainerItems; import com.android.launcher3.model.QuickstepModelDelegate.PredictorState; import com.android.launcher3.model.data.ItemInfo; import com.android.launcher3.util.ComponentKey; import com.android.launcher3.util.PackageUserKey; import com.android.launcher3.widget.PendingAddWidgetInfo; import java.util.ArrayList; import java.util.List; import java.util.Map; import java.util.Set; import java.util.function.Predicate; import java.util.stream.Collectors; /** Task to update model as a result of predicted widgets update */ public final class WidgetsPredictionUpdateTask extends BaseModelUpdateTask { private final PredictorState mPredictorState; private final List mTargets; WidgetsPredictionUpdateTask(PredictorState predictorState, List targets) { mPredictorState = predictorState; mTargets = targets; } /** * Uses the app predication result to infer widgets that the user may want to use. * *

The algorithm uses the app prediction ranking to create a widgets ranking which only * includes one widget per app and excludes widgets that have already been added to the * workspace. */ @Override public void execute(@NonNull final LauncherAppState appState, @NonNull final BgDataModel dataModel, @NonNull final AllAppsList apps) { Set widgetsInWorkspace = dataModel.appWidgets.stream().map( widget -> new ComponentKey(widget.providerName, widget.user)).collect( Collectors.toSet()); Predicate notOnWorkspace = w -> !widgetsInWorkspace.contains(w); Map> allWidgets = dataModel.widgetsModel.getAllWidgetsWithoutShortcuts(); List servicePredictedItems = new ArrayList<>(); List localFilteredWidgets = new ArrayList<>(); for (AppTarget app : mTargets) { PackageUserKey packageUserKey = new PackageUserKey(app.getPackageName(), app.getUser()); List widgets = allWidgets.get(packageUserKey); if (widgets == null || widgets.isEmpty()) { continue; } String className = app.getClassName(); if (!TextUtils.isEmpty(className)) { WidgetItem item = widgets.stream() .filter(w -> className.equals(w.componentName.getClassName())) .filter(notOnWorkspace) .findFirst() .orElse(null); if (item != null) { servicePredictedItems.add(item); continue; } } // No widget was added by the service, try local filtering widgets.stream().filter(notOnWorkspace).findFirst() .ifPresent(localFilteredWidgets::add); } if (servicePredictedItems.isEmpty()) { servicePredictedItems.addAll(localFilteredWidgets); } List items = servicePredictedItems.stream() .map(it -> new PendingAddWidgetInfo(it.widgetInfo, CONTAINER_WIDGETS_PREDICTION)) .collect(Collectors.toList()); FixedContainerItems fixedContainerItems = new FixedContainerItems(mPredictorState.containerId, items); dataModel.extraItems.put(mPredictorState.containerId, fixedContainerItems); bindExtraContainerItems(fixedContainerItems); // Don't store widgets prediction to disk because it is not used frequently. } }