1cc100844c
- Add a launcher "system shortcut" to bubble deep shortcuts or apps
- For shortcuts, save the shortcut info to use to produce the bubble
- Add an interface so that the system shortcut can call through to
sysui proxy
Flag: com.android.wm.shell.enable_bubble_anything
Test: manual - enable the flag and try to bubble a shortcut via the
longpress menu on launcher
- try bubbling an app on launcher
- try bubbling an app from the taskbar
Bug: 342245211
Change-Id: I1ef49e1628ba0be9cea05073ecd9cd66bf67f88f
255 lines
8.4 KiB
Java
255 lines
8.4 KiB
Java
/*
|
|
* Copyright (C) 2008 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.data;
|
|
|
|
import android.app.Person;
|
|
import android.content.ComponentName;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.pm.ShortcutInfo;
|
|
import android.text.TextUtils;
|
|
import android.util.Log;
|
|
|
|
import androidx.annotation.NonNull;
|
|
import androidx.annotation.Nullable;
|
|
|
|
import com.android.launcher3.Flags;
|
|
import com.android.launcher3.LauncherSettings;
|
|
import com.android.launcher3.LauncherSettings.Favorites;
|
|
import com.android.launcher3.Utilities;
|
|
import com.android.launcher3.icons.IconCache;
|
|
import com.android.launcher3.pm.UserCache;
|
|
import com.android.launcher3.shortcuts.ShortcutKey;
|
|
import com.android.launcher3.util.ApiWrapper;
|
|
import com.android.launcher3.util.ContentWriter;
|
|
|
|
import java.util.Arrays;
|
|
|
|
/**
|
|
* Represents a launchable icon on the workspaces and in folders.
|
|
*/
|
|
public class WorkspaceItemInfo extends ItemInfoWithIcon {
|
|
|
|
public static final int DEFAULT = 0;
|
|
|
|
/**
|
|
* The shortcut was restored from a backup and it not ready to be used. This is automatically
|
|
* set during backup/restore
|
|
*/
|
|
public static final int FLAG_RESTORED_ICON = 1;
|
|
|
|
/**
|
|
* The icon was added as an auto-install app, and is not ready to be used. This flag can't
|
|
* be present along with {@link #FLAG_RESTORED_ICON}, and is set during default layout
|
|
* parsing.
|
|
*
|
|
* OR this icon was added due to it being an active install session created by the user.
|
|
*/
|
|
public static final int FLAG_AUTOINSTALL_ICON = 1 << 1;
|
|
|
|
/**
|
|
* Indicates that the widget restore has started.
|
|
*/
|
|
public static final int FLAG_RESTORE_STARTED = 1 << 2;
|
|
|
|
/**
|
|
* Web UI supported.
|
|
*/
|
|
public static final int FLAG_SUPPORTS_WEB_UI = 1 << 3;
|
|
|
|
/**
|
|
*
|
|
*/
|
|
public static final int FLAG_START_FOR_RESULT = 1 << 4;
|
|
|
|
/**
|
|
* The intent used to start the application.
|
|
*/
|
|
@NonNull
|
|
public Intent intent;
|
|
|
|
/**
|
|
* A message to display when the user tries to start a disabled shortcut.
|
|
* This is currently only used for deep shortcuts.
|
|
*/
|
|
public CharSequence disabledMessage;
|
|
|
|
public int status;
|
|
|
|
/**
|
|
* A set of person's Id associated with the WorkspaceItemInfo, this is only used if the item
|
|
* represents a deep shortcut.
|
|
*/
|
|
@NonNull private String[] personKeys = Utilities.EMPTY_STRING_ARRAY;
|
|
|
|
public int options;
|
|
|
|
@Nullable
|
|
private ShortcutInfo mShortcutInfo = null;
|
|
|
|
public WorkspaceItemInfo() {
|
|
itemType = LauncherSettings.Favorites.ITEM_TYPE_APPLICATION;
|
|
}
|
|
|
|
public WorkspaceItemInfo(WorkspaceItemInfo info) {
|
|
super(info);
|
|
title = info.title;
|
|
intent = new Intent(info.intent);
|
|
status = info.status;
|
|
personKeys = info.personKeys.clone();
|
|
}
|
|
|
|
/** TODO: Remove this. It's only called by ApplicationInfo.makeWorkspaceItem. */
|
|
public WorkspaceItemInfo(AppInfo info) {
|
|
super(info);
|
|
title = Utilities.trim(info.title);
|
|
intent = new Intent(info.getIntent());
|
|
}
|
|
|
|
/**
|
|
* Creates a {@link WorkspaceItemInfo} from a {@link ShortcutInfo}.
|
|
*/
|
|
public WorkspaceItemInfo(ShortcutInfo shortcutInfo, Context context) {
|
|
user = shortcutInfo.getUserHandle();
|
|
itemType = Favorites.ITEM_TYPE_DEEP_SHORTCUT;
|
|
if (Flags.privateSpaceRestrictAccessibilityDrag()) {
|
|
if (UserCache.INSTANCE.get(context).getUserInfo(user).isPrivate()) {
|
|
runtimeStatusFlags |= FLAG_NOT_PINNABLE;
|
|
}
|
|
}
|
|
updateFromDeepShortcutInfo(shortcutInfo, context);
|
|
}
|
|
|
|
@Override
|
|
public void onAddToDatabase(@NonNull ContentWriter writer) {
|
|
super.onAddToDatabase(writer);
|
|
writer.put(Favorites.TITLE, title)
|
|
.put(Favorites.INTENT, getIntent())
|
|
.put(Favorites.OPTIONS, options)
|
|
.put(Favorites.RESTORED, status);
|
|
|
|
if (!usingLowResIcon()) {
|
|
writer.putIcon(bitmap, user);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
@NonNull
|
|
public Intent getIntent() {
|
|
return intent;
|
|
}
|
|
|
|
public boolean hasStatusFlag(int flag) {
|
|
return (status & flag) != 0;
|
|
}
|
|
|
|
|
|
public final boolean isPromise() {
|
|
return hasStatusFlag(FLAG_RESTORED_ICON | FLAG_AUTOINSTALL_ICON)
|
|
// For archived apps, promise icons are always ready to be displayed.
|
|
|| isArchived();
|
|
}
|
|
|
|
/**
|
|
* Returns true if the workspace item supports promise icon UI. There are a few cases where they
|
|
* are supported:
|
|
* 1. Icons to be restored via backup/restore.
|
|
* 2. Icons added as an auto-install app.
|
|
* 3. Icons added due to it being an active install session created by the user.
|
|
* 4. Icons for archived apps.
|
|
*/
|
|
public boolean hasPromiseIconUi() {
|
|
return isPromise() && !hasStatusFlag(FLAG_SUPPORTS_WEB_UI);
|
|
}
|
|
|
|
public void updateFromDeepShortcutInfo(@NonNull final ShortcutInfo shortcutInfo,
|
|
@NonNull final Context context) {
|
|
if (com.android.wm.shell.Flags.enableBubbleAnything()) {
|
|
mShortcutInfo = shortcutInfo;
|
|
}
|
|
// {@link ShortcutInfo#getActivity} can change during an update. Recreate the intent
|
|
intent = ShortcutKey.makeIntent(shortcutInfo);
|
|
title = shortcutInfo.getShortLabel();
|
|
|
|
CharSequence label = shortcutInfo.getLongLabel();
|
|
if (TextUtils.isEmpty(label)) {
|
|
label = shortcutInfo.getShortLabel();
|
|
}
|
|
contentDescription = context.getPackageManager().getUserBadgedLabel(label, user);
|
|
if (shortcutInfo.isEnabled()) {
|
|
runtimeStatusFlags &= ~FLAG_DISABLED_BY_PUBLISHER;
|
|
} else {
|
|
Log.w(TAG, "updateFromDeepShortcutInfo: Updated shortcut has been disabled. "
|
|
+ " package=" + shortcutInfo.getPackage()
|
|
+ " disabledReason=" + shortcutInfo.getDisabledReason());
|
|
runtimeStatusFlags |= FLAG_DISABLED_BY_PUBLISHER;
|
|
}
|
|
|
|
if (shortcutInfo.getDisabledReason() == ShortcutInfo.DISABLED_REASON_VERSION_LOWER) {
|
|
runtimeStatusFlags |= FLAG_DISABLED_VERSION_LOWER;
|
|
} else {
|
|
runtimeStatusFlags &= ~FLAG_DISABLED_VERSION_LOWER;
|
|
}
|
|
|
|
Person[] persons = ApiWrapper.INSTANCE.get(context).getPersons(shortcutInfo);
|
|
personKeys = persons.length == 0 ? Utilities.EMPTY_STRING_ARRAY
|
|
: Arrays.stream(persons).map(Person::getKey).sorted().toArray(String[]::new);
|
|
}
|
|
|
|
@Nullable
|
|
public ShortcutInfo getDeepShortcutInfo() {
|
|
return mShortcutInfo;
|
|
}
|
|
|
|
/**
|
|
* {@code true} if the shortcut is disabled due to its app being a lower version.
|
|
*/
|
|
public boolean isDisabledVersionLower() {
|
|
return (runtimeStatusFlags & FLAG_DISABLED_VERSION_LOWER) != 0;
|
|
}
|
|
|
|
/** Returns the WorkspaceItemInfo id associated with the deep shortcut. */
|
|
public String getDeepShortcutId() {
|
|
return itemType == Favorites.ITEM_TYPE_DEEP_SHORTCUT
|
|
? getIntent().getStringExtra(ShortcutKey.EXTRA_SHORTCUT_ID) : null;
|
|
}
|
|
|
|
@NonNull
|
|
public String[] getPersonKeys() {
|
|
return personKeys;
|
|
}
|
|
|
|
@Override
|
|
public ComponentName getTargetComponent() {
|
|
ComponentName cn = super.getTargetComponent();
|
|
if (cn == null && hasStatusFlag(
|
|
FLAG_SUPPORTS_WEB_UI | FLAG_AUTOINSTALL_ICON | FLAG_RESTORED_ICON)) {
|
|
// Legacy shortcuts and promise icons with web UI may not have a componentName but just
|
|
// a packageName. In that case create a empty componentName instead of adding additional
|
|
// check everywhere.
|
|
String pkg = intent.getPackage();
|
|
return pkg == null ? null : new ComponentName(pkg, IconCache.EMPTY_CLASS_NAME);
|
|
}
|
|
return cn;
|
|
}
|
|
|
|
@Override
|
|
public WorkspaceItemInfo clone() {
|
|
return new WorkspaceItemInfo(this);
|
|
}
|
|
}
|