mirror of
https://github.com/LawnchairLauncher/lawnchair.git
synced 2026-02-17 17:48:20 +00:00
Bug: 242669604 Test: Verified that the RPC is not made on UI thread Change-Id: I4299628b8b3e9aa2844f5556c57f47d997dd85e1
364 lines
16 KiB
Java
364 lines
16 KiB
Java
/*
|
|
* Copyright (C) 2018 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.touch;
|
|
|
|
import static com.android.launcher3.Launcher.REQUEST_BIND_PENDING_APPWIDGET;
|
|
import static com.android.launcher3.Launcher.REQUEST_RECONFIGURE_APPWIDGET;
|
|
import static com.android.launcher3.logging.StatsLogManager.LauncherEvent.LAUNCHER_FOLDER_OPEN;
|
|
import static com.android.launcher3.model.data.ItemInfoWithIcon.FLAG_DISABLED_BY_PUBLISHER;
|
|
import static com.android.launcher3.model.data.ItemInfoWithIcon.FLAG_DISABLED_LOCKED_USER;
|
|
import static com.android.launcher3.model.data.ItemInfoWithIcon.FLAG_DISABLED_QUIET_USER;
|
|
import static com.android.launcher3.model.data.ItemInfoWithIcon.FLAG_DISABLED_SAFEMODE;
|
|
import static com.android.launcher3.model.data.ItemInfoWithIcon.FLAG_DISABLED_SUSPENDED;
|
|
import static com.android.launcher3.util.Executors.MAIN_EXECUTOR;
|
|
import static com.android.launcher3.util.Executors.UI_HELPER_EXECUTOR;
|
|
|
|
import android.app.AlertDialog;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.pm.LauncherApps;
|
|
import android.content.pm.PackageInstaller.SessionInfo;
|
|
import android.text.TextUtils;
|
|
import android.util.Log;
|
|
import android.view.View;
|
|
import android.view.View.OnClickListener;
|
|
import android.widget.Toast;
|
|
|
|
import com.android.launcher3.BubbleTextView;
|
|
import com.android.launcher3.Launcher;
|
|
import com.android.launcher3.LauncherSettings;
|
|
import com.android.launcher3.R;
|
|
import com.android.launcher3.Utilities;
|
|
import com.android.launcher3.folder.Folder;
|
|
import com.android.launcher3.folder.FolderIcon;
|
|
import com.android.launcher3.logging.InstanceId;
|
|
import com.android.launcher3.logging.InstanceIdSequence;
|
|
import com.android.launcher3.logging.StatsLogManager;
|
|
import com.android.launcher3.model.data.AppInfo;
|
|
import com.android.launcher3.model.data.FolderInfo;
|
|
import com.android.launcher3.model.data.ItemInfo;
|
|
import com.android.launcher3.model.data.ItemInfoWithIcon;
|
|
import com.android.launcher3.model.data.LauncherAppWidgetInfo;
|
|
import com.android.launcher3.model.data.WorkspaceItemInfo;
|
|
import com.android.launcher3.pm.InstallSessionHelper;
|
|
import com.android.launcher3.shortcuts.ShortcutKey;
|
|
import com.android.launcher3.testing.TestLogging;
|
|
import com.android.launcher3.testing.shared.TestProtocol;
|
|
import com.android.launcher3.util.ItemInfoMatcher;
|
|
import com.android.launcher3.util.PackageManagerHelper;
|
|
import com.android.launcher3.views.FloatingIconView;
|
|
import com.android.launcher3.widget.LauncherAppWidgetProviderInfo;
|
|
import com.android.launcher3.widget.PendingAppWidgetHostView;
|
|
import com.android.launcher3.widget.WidgetAddFlowHandler;
|
|
import com.android.launcher3.widget.WidgetManagerHelper;
|
|
|
|
import java.util.Collections;
|
|
import java.util.concurrent.CompletableFuture;
|
|
import java.util.function.Consumer;
|
|
|
|
/**
|
|
* Class for handling clicks on workspace and all-apps items
|
|
*/
|
|
public class ItemClickHandler {
|
|
|
|
private static final String TAG = ItemClickHandler.class.getSimpleName();
|
|
|
|
/**
|
|
* Instance used for click handling on items
|
|
*/
|
|
public static final OnClickListener INSTANCE = ItemClickHandler::onClick;
|
|
|
|
private static void onClick(View v) {
|
|
// Make sure that rogue clicks don't get through while allapps is launching, or after the
|
|
// view has detached (it's possible for this to happen if the view is removed mid touch).
|
|
if (v.getWindowToken() == null) return;
|
|
|
|
Launcher launcher = Launcher.getLauncher(v.getContext());
|
|
if (!launcher.getWorkspace().isFinishedSwitchingState()) return;
|
|
|
|
Object tag = v.getTag();
|
|
if (tag instanceof WorkspaceItemInfo) {
|
|
onClickAppShortcut(v, (WorkspaceItemInfo) tag, launcher);
|
|
} else if (tag instanceof FolderInfo) {
|
|
if (v instanceof FolderIcon) {
|
|
onClickFolderIcon(v);
|
|
}
|
|
} else if (tag instanceof AppInfo) {
|
|
startAppShortcutOrInfoActivity(v, (AppInfo) tag, launcher);
|
|
} else if (tag instanceof LauncherAppWidgetInfo) {
|
|
if (v instanceof PendingAppWidgetHostView) {
|
|
onClickPendingWidget((PendingAppWidgetHostView) v, launcher);
|
|
}
|
|
} else if (tag instanceof ItemClickProxy) {
|
|
((ItemClickProxy) tag).onItemClicked(v);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Event handler for a folder icon click.
|
|
*
|
|
* @param v The view that was clicked. Must be an instance of {@link FolderIcon}.
|
|
*/
|
|
private static void onClickFolderIcon(View v) {
|
|
Folder folder = ((FolderIcon) v).getFolder();
|
|
if (!folder.isOpen() && !folder.isDestroyed()) {
|
|
// Open the requested folder
|
|
folder.animateOpen();
|
|
StatsLogManager.newInstance(v.getContext()).logger().withItemInfo(folder.mInfo)
|
|
.log(LAUNCHER_FOLDER_OPEN);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Event handler for the app widget view which has not fully restored.
|
|
*/
|
|
private static void onClickPendingWidget(PendingAppWidgetHostView v, Launcher launcher) {
|
|
if (launcher.getPackageManager().isSafeMode()) {
|
|
Toast.makeText(launcher, R.string.safemode_widget_error, Toast.LENGTH_SHORT).show();
|
|
return;
|
|
}
|
|
|
|
final LauncherAppWidgetInfo info = (LauncherAppWidgetInfo) v.getTag();
|
|
if (v.isReadyForClickSetup()) {
|
|
LauncherAppWidgetProviderInfo appWidgetInfo = new WidgetManagerHelper(launcher)
|
|
.findProvider(info.providerName, info.user);
|
|
if (appWidgetInfo == null) {
|
|
return;
|
|
}
|
|
WidgetAddFlowHandler addFlowHandler = new WidgetAddFlowHandler(appWidgetInfo);
|
|
|
|
if (info.hasRestoreFlag(LauncherAppWidgetInfo.FLAG_ID_NOT_VALID)) {
|
|
if (!info.hasRestoreFlag(LauncherAppWidgetInfo.FLAG_ID_ALLOCATED)) {
|
|
// This should not happen, as we make sure that an Id is allocated during bind.
|
|
return;
|
|
}
|
|
addFlowHandler.startBindFlow(launcher, info.appWidgetId, info,
|
|
REQUEST_BIND_PENDING_APPWIDGET);
|
|
} else {
|
|
addFlowHandler.startConfigActivity(launcher, info, REQUEST_RECONFIGURE_APPWIDGET);
|
|
}
|
|
} else {
|
|
final String packageName = info.providerName.getPackageName();
|
|
onClickPendingAppItem(v, launcher, packageName, info.installProgress >= 0);
|
|
}
|
|
}
|
|
|
|
private static void onClickPendingAppItem(View v, Launcher launcher, String packageName,
|
|
boolean downloadStarted) {
|
|
ItemInfo item = (ItemInfo) v.getTag();
|
|
CompletableFuture<SessionInfo> siFuture;
|
|
if (Utilities.ATLEAST_Q) {
|
|
siFuture = CompletableFuture.supplyAsync(() ->
|
|
InstallSessionHelper.INSTANCE.get(launcher)
|
|
.getActiveSessionInfo(item.user, packageName),
|
|
UI_HELPER_EXECUTOR);
|
|
} else {
|
|
siFuture = CompletableFuture.completedFuture(null);
|
|
}
|
|
Consumer<SessionInfo> marketLaunchAction = sessionInfo -> {
|
|
if (sessionInfo != null && Utilities.ATLEAST_Q) {
|
|
LauncherApps launcherApps = launcher.getSystemService(LauncherApps.class);
|
|
try {
|
|
launcherApps.startPackageInstallerSessionDetailsActivity(sessionInfo, null,
|
|
launcher.getActivityLaunchOptions(v, item).toBundle());
|
|
return;
|
|
} catch (Exception e) {
|
|
Log.e(TAG, "Unable to launch market intent for package=" + packageName, e);
|
|
}
|
|
}
|
|
// Fallback to using custom market intent.
|
|
Intent intent = new PackageManagerHelper(launcher).getMarketIntent(packageName);
|
|
launcher.startActivitySafely(v, intent, item);
|
|
};
|
|
|
|
if (downloadStarted) {
|
|
// If the download has started, simply direct to the market app.
|
|
siFuture.thenAcceptAsync(marketLaunchAction, MAIN_EXECUTOR);
|
|
return;
|
|
}
|
|
new AlertDialog.Builder(launcher)
|
|
.setTitle(R.string.abandoned_promises_title)
|
|
.setMessage(R.string.abandoned_promise_explanation)
|
|
.setPositiveButton(R.string.abandoned_search,
|
|
(d, i) -> siFuture.thenAcceptAsync(marketLaunchAction, MAIN_EXECUTOR))
|
|
.setNeutralButton(R.string.abandoned_clean_this,
|
|
(d, i) -> launcher.getWorkspace()
|
|
.persistRemoveItemsByMatcher(ItemInfoMatcher.ofPackages(
|
|
Collections.singleton(packageName), item.user),
|
|
"user explicitly removes the promise app icon"))
|
|
.create().show();
|
|
}
|
|
|
|
/**
|
|
* Handles clicking on a disabled shortcut
|
|
*
|
|
* @return true iff the disabled item click has been handled.
|
|
*/
|
|
public static boolean handleDisabledItemClicked(WorkspaceItemInfo shortcut, Context context) {
|
|
final int disabledFlags = shortcut.runtimeStatusFlags
|
|
& WorkspaceItemInfo.FLAG_DISABLED_MASK;
|
|
// Handle the case where the disabled reason is DISABLED_REASON_VERSION_LOWER.
|
|
// Show an AlertDialog for the user to choose either updating the app or cancel the launch.
|
|
if (maybeCreateAlertDialogForShortcut(shortcut, context)) {
|
|
return true;
|
|
}
|
|
|
|
if ((disabledFlags
|
|
& ~FLAG_DISABLED_SUSPENDED
|
|
& ~FLAG_DISABLED_QUIET_USER) == 0) {
|
|
// If the app is only disabled because of the above flags, launch activity anyway.
|
|
// Framework will tell the user why the app is suspended.
|
|
return false;
|
|
} else {
|
|
if (!TextUtils.isEmpty(shortcut.disabledMessage)) {
|
|
// Use a message specific to this shortcut, if it has one.
|
|
Toast.makeText(context, shortcut.disabledMessage, Toast.LENGTH_SHORT).show();
|
|
return true;
|
|
}
|
|
// Otherwise just use a generic error message.
|
|
int error = R.string.activity_not_available;
|
|
if ((shortcut.runtimeStatusFlags & FLAG_DISABLED_SAFEMODE) != 0) {
|
|
error = R.string.safemode_shortcut_error;
|
|
} else if ((shortcut.runtimeStatusFlags & FLAG_DISABLED_BY_PUBLISHER) != 0
|
|
|| (shortcut.runtimeStatusFlags & FLAG_DISABLED_LOCKED_USER) != 0) {
|
|
error = R.string.shortcut_not_available;
|
|
}
|
|
Toast.makeText(context, error, Toast.LENGTH_SHORT).show();
|
|
return true;
|
|
}
|
|
}
|
|
|
|
private static boolean maybeCreateAlertDialogForShortcut(final WorkspaceItemInfo shortcut,
|
|
Context context) {
|
|
try {
|
|
final Launcher launcher = Launcher.getLauncher(context);
|
|
if (shortcut.itemType == LauncherSettings.Favorites.ITEM_TYPE_DEEP_SHORTCUT
|
|
&& shortcut.isDisabledVersionLower()) {
|
|
final Intent marketIntent = shortcut.getMarketIntent(context);
|
|
// No market intent means no target package for the shortcut, which should be an
|
|
// issue. Falling back to showing toast messages.
|
|
if (marketIntent == null) {
|
|
return false;
|
|
}
|
|
|
|
new AlertDialog.Builder(context)
|
|
.setTitle(R.string.dialog_update_title)
|
|
.setMessage(R.string.dialog_update_message)
|
|
.setPositiveButton(R.string.dialog_update, (d, i) -> {
|
|
// Direct the user to the play store to update the app
|
|
context.startActivity(marketIntent);
|
|
})
|
|
.setNeutralButton(R.string.dialog_remove, (d, i) -> {
|
|
// Remove the icon if launcher is successfully initialized
|
|
launcher.getWorkspace().persistRemoveItemsByMatcher(ItemInfoMatcher
|
|
.ofShortcutKeys(Collections.singleton(ShortcutKey
|
|
.fromItemInfo(shortcut))),
|
|
"user explicitly removes disabled shortcut");
|
|
})
|
|
.create()
|
|
.show();
|
|
return true;
|
|
}
|
|
} catch (Exception e) {
|
|
Log.e(TAG, "Error creating alert dialog", e);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Event handler for an app shortcut click.
|
|
*
|
|
* @param v The view that was clicked. Must be a tagged with a {@link WorkspaceItemInfo}.
|
|
*/
|
|
public static void onClickAppShortcut(View v, WorkspaceItemInfo shortcut, Launcher launcher) {
|
|
if (shortcut.isDisabled() && handleDisabledItemClicked(shortcut, launcher)) {
|
|
return;
|
|
}
|
|
|
|
// Check for abandoned promise
|
|
if ((v instanceof BubbleTextView) && shortcut.hasPromiseIconUi()) {
|
|
String packageName = shortcut.getIntent().getComponent() != null
|
|
? shortcut.getIntent().getComponent().getPackageName()
|
|
: shortcut.getIntent().getPackage();
|
|
if (!TextUtils.isEmpty(packageName)) {
|
|
onClickPendingAppItem(
|
|
v,
|
|
launcher,
|
|
packageName,
|
|
(shortcut.runtimeStatusFlags
|
|
& ItemInfoWithIcon.FLAG_INSTALL_SESSION_ACTIVE) != 0);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Start activities
|
|
startAppShortcutOrInfoActivity(v, shortcut, launcher);
|
|
}
|
|
|
|
private static void startAppShortcutOrInfoActivity(View v, ItemInfo item, Launcher launcher) {
|
|
TestLogging.recordEvent(
|
|
TestProtocol.SEQUENCE_MAIN, "start: startAppShortcutOrInfoActivity");
|
|
Intent intent;
|
|
if (item instanceof ItemInfoWithIcon
|
|
&& (((ItemInfoWithIcon) item).runtimeStatusFlags
|
|
& ItemInfoWithIcon.FLAG_INSTALL_SESSION_ACTIVE) != 0) {
|
|
ItemInfoWithIcon appInfo = (ItemInfoWithIcon) item;
|
|
intent = new PackageManagerHelper(launcher)
|
|
.getMarketIntent(appInfo.getTargetComponent().getPackageName());
|
|
} else {
|
|
intent = item.getIntent();
|
|
}
|
|
if (intent == null) {
|
|
throw new IllegalArgumentException("Input must have a valid intent");
|
|
}
|
|
if (item instanceof WorkspaceItemInfo) {
|
|
WorkspaceItemInfo si = (WorkspaceItemInfo) item;
|
|
if (si.hasStatusFlag(WorkspaceItemInfo.FLAG_SUPPORTS_WEB_UI)
|
|
&& Intent.ACTION_VIEW.equals(intent.getAction())) {
|
|
// make a copy of the intent that has the package set to null
|
|
// we do this because the platform sometimes disables instant
|
|
// apps temporarily (triggered by the user) and fallbacks to the
|
|
// web ui. This only works though if the package isn't set
|
|
intent = new Intent(intent);
|
|
intent.setPackage(null);
|
|
}
|
|
if ((si.options & WorkspaceItemInfo.FLAG_START_FOR_RESULT) != 0) {
|
|
launcher.startActivityForResult(item.getIntent(), 0);
|
|
InstanceId instanceId = new InstanceIdSequence().newInstanceId();
|
|
launcher.logAppLaunch(launcher.getStatsLogManager(), item, instanceId);
|
|
return;
|
|
}
|
|
}
|
|
if (v != null && launcher.supportsAdaptiveIconAnimation(v)) {
|
|
// Preload the icon to reduce latency b/w swapping the floating view with the original.
|
|
FloatingIconView.fetchIcon(launcher, v, item, true /* isOpening */);
|
|
}
|
|
launcher.startActivitySafely(v, intent, item);
|
|
}
|
|
|
|
/**
|
|
* Interface to indicate that an item will handle the click itself.
|
|
*/
|
|
public interface ItemClickProxy {
|
|
|
|
/**
|
|
* Called when the item is clicked
|
|
*/
|
|
void onItemClicked(View view);
|
|
}
|
|
}
|