/*
|
* 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;
|
|
import android.appwidget.AppWidgetManager;
|
import android.appwidget.AppWidgetProviderInfo;
|
import android.content.BroadcastReceiver;
|
import android.content.Context;
|
import android.content.Intent;
|
import android.content.SharedPreferences;
|
import android.content.pm.ActivityInfo;
|
import android.content.pm.LauncherActivityInfo;
|
import android.content.pm.PackageManager;
|
import android.content.pm.ShortcutInfo;
|
import android.graphics.Bitmap;
|
import android.graphics.BitmapFactory;
|
import android.os.Handler;
|
import android.os.Looper;
|
import android.os.Message;
|
import android.os.Parcelable;
|
import android.os.Process;
|
import android.os.UserHandle;
|
import android.text.TextUtils;
|
import android.util.Base64;
|
import android.util.Log;
|
import android.util.Pair;
|
|
import com.android.launcher3.compat.LauncherAppsCompat;
|
import com.android.launcher3.compat.UserManagerCompat;
|
import com.android.launcher3.icons.BitmapInfo;
|
import com.android.launcher3.icons.GraphicsUtils;
|
import com.android.launcher3.icons.LauncherIcons;
|
import com.android.launcher3.shortcuts.DeepShortcutManager;
|
import com.android.launcher3.shortcuts.ShortcutKey;
|
import com.android.launcher3.util.PackageManagerHelper;
|
import com.android.launcher3.util.Preconditions;
|
import com.android.launcher3.util.Thunk;
|
|
import org.json.JSONException;
|
import org.json.JSONObject;
|
import org.json.JSONStringer;
|
|
import java.net.URISyntaxException;
|
import java.util.ArrayList;
|
import java.util.Arrays;
|
import java.util.Collection;
|
import java.util.HashSet;
|
import java.util.Iterator;
|
import java.util.List;
|
import java.util.Set;
|
|
public class InstallShortcutReceiver extends BroadcastReceiver {
|
|
private static final int MSG_ADD_TO_QUEUE = 1;
|
private static final int MSG_FLUSH_QUEUE = 2;
|
|
public static final int FLAG_ACTIVITY_PAUSED = 1;
|
public static final int FLAG_LOADER_RUNNING = 2;
|
public static final int FLAG_DRAG_AND_DROP = 4;
|
public static final int FLAG_BULK_ADD = 4;
|
|
// Determines whether to defer installing shortcuts immediately until
|
// processAllPendingInstalls() is called.
|
private static int sInstallQueueDisabledFlags = 0;
|
|
private static final String TAG = "InstallShortcutReceiver";
|
private static final boolean DBG = false;
|
|
private static final String ACTION_INSTALL_SHORTCUT =
|
"com.android.launcher.action.INSTALL_SHORTCUT";
|
|
private static final String LAUNCH_INTENT_KEY = "intent.launch";
|
private static final String NAME_KEY = "name";
|
private static final String ICON_KEY = "icon";
|
private static final String ICON_RESOURCE_NAME_KEY = "iconResource";
|
private static final String ICON_RESOURCE_PACKAGE_NAME_KEY = "iconResourcePackage";
|
|
private static final String APP_SHORTCUT_TYPE_KEY = "isAppShortcut";
|
private static final String DEEPSHORTCUT_TYPE_KEY = "isDeepShortcut";
|
private static final String APP_WIDGET_TYPE_KEY = "isAppWidget";
|
private static final String USER_HANDLE_KEY = "userHandle";
|
|
// The set of shortcuts that are pending install
|
private static final String APPS_PENDING_INSTALL = "apps_to_install";
|
|
public static final int NEW_SHORTCUT_BOUNCE_DURATION = 450;
|
public static final int NEW_SHORTCUT_STAGGER_DELAY = 85;
|
|
private static final Handler sHandler = new Handler(LauncherModel.getWorkerLooper()) {
|
|
@Override
|
public void handleMessage(Message msg) {
|
switch (msg.what) {
|
case MSG_ADD_TO_QUEUE: {
|
Pair<Context, PendingInstallShortcutInfo> pair =
|
(Pair<Context, PendingInstallShortcutInfo>) msg.obj;
|
String encoded = pair.second.encodeToString();
|
SharedPreferences prefs = Utilities.getPrefs(pair.first);
|
Set<String> strings = prefs.getStringSet(APPS_PENDING_INSTALL, null);
|
strings = (strings != null) ? new HashSet<>(strings) : new HashSet<String>(1);
|
strings.add(encoded);
|
prefs.edit().putStringSet(APPS_PENDING_INSTALL, strings).apply();
|
return;
|
}
|
case MSG_FLUSH_QUEUE: {
|
Context context = (Context) msg.obj;
|
LauncherModel model = LauncherAppState.getInstance(context).getModel();
|
if (model.getCallback() == null) {
|
// Launcher not loaded
|
return;
|
}
|
|
ArrayList<Pair<ItemInfo, Object>> installQueue = new ArrayList<>();
|
SharedPreferences prefs = Utilities.getPrefs(context);
|
Set<String> strings = prefs.getStringSet(APPS_PENDING_INSTALL, null);
|
if (DBG) Log.d(TAG, "Getting and clearing APPS_PENDING_INSTALL: " + strings);
|
if (strings == null) {
|
return;
|
}
|
|
LauncherAppsCompat launcherApps = LauncherAppsCompat.getInstance(context);
|
for (String encoded : strings) {
|
PendingInstallShortcutInfo info = decode(encoded, context);
|
if (info == null) {
|
continue;
|
}
|
|
String pkg = getIntentPackage(info.launchIntent);
|
if (!TextUtils.isEmpty(pkg)
|
&& !launcherApps.isPackageEnabledForProfile(pkg, info.user)) {
|
if (DBG) Log.d(TAG, "Ignoring shortcut for absent package: "
|
+ info.launchIntent);
|
continue;
|
}
|
|
// Generate a shortcut info to add into the model
|
installQueue.add(info.getItemInfo());
|
}
|
prefs.edit().remove(APPS_PENDING_INSTALL).apply();
|
if (!installQueue.isEmpty()) {
|
model.addAndBindAddedWorkspaceItems(installQueue);
|
}
|
return;
|
}
|
}
|
}
|
};
|
|
public static void removeFromInstallQueue(Context context, HashSet<String> packageNames,
|
UserHandle user) {
|
if (packageNames.isEmpty()) {
|
return;
|
}
|
Preconditions.assertWorkerThread();
|
|
SharedPreferences sp = Utilities.getPrefs(context);
|
Set<String> strings = sp.getStringSet(APPS_PENDING_INSTALL, null);
|
if (DBG) {
|
Log.d(TAG, "APPS_PENDING_INSTALL: " + strings
|
+ ", removing packages: " + packageNames);
|
}
|
if (strings == null || ((Collection) strings).isEmpty()) {
|
return;
|
}
|
Set<String> newStrings = new HashSet<>(strings);
|
Iterator<String> newStringsIter = newStrings.iterator();
|
while (newStringsIter.hasNext()) {
|
String encoded = newStringsIter.next();
|
try {
|
Decoder decoder = new Decoder(encoded, context);
|
if (packageNames.contains(getIntentPackage(decoder.launcherIntent)) &&
|
user.equals(decoder.user)) {
|
newStringsIter.remove();
|
}
|
} catch (JSONException | URISyntaxException e) {
|
Log.d(TAG, "Exception reading shortcut to add: " + e);
|
newStringsIter.remove();
|
}
|
}
|
sp.edit().putStringSet(APPS_PENDING_INSTALL, newStrings).apply();
|
}
|
|
public void onReceive(Context context, Intent data) {
|
if (!ACTION_INSTALL_SHORTCUT.equals(data.getAction())) {
|
return;
|
}
|
PendingInstallShortcutInfo info = createPendingInfo(context, data);
|
if (info != null) {
|
if (!info.isLauncherActivity()) {
|
// Since its a custom shortcut, verify that it is safe to launch.
|
if (!new PackageManagerHelper(context).hasPermissionForActivity(
|
info.launchIntent, null)) {
|
// Target cannot be launched, or requires some special permission to launch
|
Log.e(TAG, "Ignoring malicious intent " + info.launchIntent.toUri(0));
|
return;
|
}
|
}
|
queuePendingShortcutInfo(info, context);
|
}
|
}
|
|
/**
|
* @return true is the extra is either null or is of type {@param type}
|
*/
|
private static boolean isValidExtraType(Intent intent, String key, Class type) {
|
Object extra = intent.getParcelableExtra(key);
|
return extra == null || type.isInstance(extra);
|
}
|
|
/**
|
* Verifies the intent and creates a {@link PendingInstallShortcutInfo}
|
*/
|
private static PendingInstallShortcutInfo createPendingInfo(Context context, Intent data) {
|
if (!isValidExtraType(data, Intent.EXTRA_SHORTCUT_INTENT, Intent.class) ||
|
!(isValidExtraType(data, Intent.EXTRA_SHORTCUT_ICON_RESOURCE,
|
Intent.ShortcutIconResource.class)) ||
|
!(isValidExtraType(data, Intent.EXTRA_SHORTCUT_ICON, Bitmap.class))) {
|
|
if (DBG) Log.e(TAG, "Invalid install shortcut intent");
|
return null;
|
}
|
|
PendingInstallShortcutInfo info = new PendingInstallShortcutInfo(
|
data, Process.myUserHandle(), context);
|
if (info.launchIntent == null || info.label == null) {
|
if (DBG) Log.e(TAG, "Invalid install shortcut intent");
|
return null;
|
}
|
|
return convertToLauncherActivityIfPossible(info);
|
}
|
|
public static WorkspaceItemInfo fromShortcutIntent(Context context, Intent data) {
|
PendingInstallShortcutInfo info = createPendingInfo(context, data);
|
return info == null ? null : (WorkspaceItemInfo) info.getItemInfo().first;
|
}
|
|
public static WorkspaceItemInfo fromActivityInfo(LauncherActivityInfo info, Context context) {
|
return (WorkspaceItemInfo) (new PendingInstallShortcutInfo(info, context).getItemInfo().first);
|
}
|
|
public static void queueShortcut(ShortcutInfo info, Context context) {
|
queuePendingShortcutInfo(new PendingInstallShortcutInfo(info, context), context);
|
}
|
|
public static void queueWidget(AppWidgetProviderInfo info, int widgetId, Context context) {
|
queuePendingShortcutInfo(new PendingInstallShortcutInfo(info, widgetId, context), context);
|
}
|
|
public static void queueActivityInfo(LauncherActivityInfo activity, Context context) {
|
queuePendingShortcutInfo(new PendingInstallShortcutInfo(activity, context), context);
|
}
|
|
public static HashSet<ShortcutKey> getPendingShortcuts(Context context) {
|
HashSet<ShortcutKey> result = new HashSet<>();
|
|
Set<String> strings = Utilities.getPrefs(context).getStringSet(APPS_PENDING_INSTALL, null);
|
if (strings == null || ((Collection) strings).isEmpty()) {
|
return result;
|
}
|
|
for (String encoded : strings) {
|
try {
|
Decoder decoder = new Decoder(encoded, context);
|
if (decoder.optBoolean(DEEPSHORTCUT_TYPE_KEY)) {
|
result.add(ShortcutKey.fromIntent(decoder.launcherIntent, decoder.user));
|
}
|
} catch (JSONException | URISyntaxException e) {
|
Log.d(TAG, "Exception reading shortcut to add: " + e);
|
}
|
}
|
return result;
|
}
|
|
private static void queuePendingShortcutInfo(PendingInstallShortcutInfo info, Context context) {
|
// Queue the item up for adding if launcher has not loaded properly yet
|
Message.obtain(sHandler, MSG_ADD_TO_QUEUE, Pair.create(context, info)).sendToTarget();
|
flushInstallQueue(context);
|
}
|
|
public static void enableInstallQueue(int flag) {
|
sInstallQueueDisabledFlags |= flag;
|
}
|
public static void disableAndFlushInstallQueue(int flag, Context context) {
|
sInstallQueueDisabledFlags &= ~flag;
|
flushInstallQueue(context);
|
}
|
|
static void flushInstallQueue(Context context) {
|
if (sInstallQueueDisabledFlags != 0) {
|
return;
|
}
|
Message.obtain(sHandler, MSG_FLUSH_QUEUE, context.getApplicationContext()).sendToTarget();
|
}
|
|
/**
|
* Ensures that we have a valid, non-null name. If the provided name is null, we will return
|
* the application name instead.
|
*/
|
@Thunk static CharSequence ensureValidName(Context context, Intent intent, CharSequence name) {
|
if (name == null) {
|
try {
|
PackageManager pm = context.getPackageManager();
|
ActivityInfo info = pm.getActivityInfo(intent.getComponent(), 0);
|
name = info.loadLabel(pm);
|
} catch (PackageManager.NameNotFoundException nnfe) {
|
return "";
|
}
|
}
|
return name;
|
}
|
|
public static class PendingInstallShortcutInfo {
|
|
final LauncherActivityInfo activityInfo;
|
final ShortcutInfo shortcutInfo;
|
final AppWidgetProviderInfo providerInfo;
|
|
final Intent data;
|
final Context mContext;
|
final Intent launchIntent;
|
final String label;
|
final UserHandle user;
|
|
/**
|
* Initializes a PendingInstallShortcutInfo received from a different app.
|
*/
|
public PendingInstallShortcutInfo(Intent data, UserHandle user, Context context) {
|
activityInfo = null;
|
shortcutInfo = null;
|
providerInfo = null;
|
|
this.data = data;
|
this.user = user;
|
mContext = context;
|
|
launchIntent = data.getParcelableExtra(Intent.EXTRA_SHORTCUT_INTENT);
|
label = data.getStringExtra(Intent.EXTRA_SHORTCUT_NAME);
|
|
}
|
|
/**
|
* Initializes a PendingInstallShortcutInfo to represent a launcher target.
|
*/
|
public PendingInstallShortcutInfo(LauncherActivityInfo info, Context context) {
|
activityInfo = info;
|
shortcutInfo = null;
|
providerInfo = null;
|
|
data = null;
|
user = info.getUser();
|
mContext = context;
|
|
launchIntent = AppInfo.makeLaunchIntent(info);
|
label = info.getLabel().toString();
|
}
|
|
/**
|
* Initializes a PendingInstallShortcutInfo to represent a launcher target.
|
*/
|
public PendingInstallShortcutInfo(ShortcutInfo info, Context context) {
|
activityInfo = null;
|
shortcutInfo = info;
|
providerInfo = null;
|
|
data = null;
|
mContext = context;
|
user = info.getUserHandle();
|
|
launchIntent = ShortcutKey.makeIntent(info);
|
label = info.getShortLabel().toString();
|
}
|
|
/**
|
* Initializes a PendingInstallShortcutInfo to represent a launcher target.
|
*/
|
public PendingInstallShortcutInfo(
|
AppWidgetProviderInfo info, int widgetId, Context context) {
|
activityInfo = null;
|
shortcutInfo = null;
|
providerInfo = info;
|
|
data = null;
|
mContext = context;
|
user = info.getProfile();
|
|
launchIntent = new Intent().setComponent(info.provider)
|
.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, widgetId);
|
label = info.label;
|
}
|
|
public String encodeToString() {
|
try {
|
if (activityInfo != null) {
|
// If it a launcher target, we only need component name, and user to
|
// recreate this.
|
return new JSONStringer()
|
.object()
|
.key(LAUNCH_INTENT_KEY).value(launchIntent.toUri(0))
|
.key(APP_SHORTCUT_TYPE_KEY).value(true)
|
.key(USER_HANDLE_KEY).value(UserManagerCompat.getInstance(mContext)
|
.getSerialNumberForUser(user))
|
.endObject().toString();
|
} else if (shortcutInfo != null) {
|
// If it a launcher target, we only need component name, and user to
|
// recreate this.
|
return new JSONStringer()
|
.object()
|
.key(LAUNCH_INTENT_KEY).value(launchIntent.toUri(0))
|
.key(DEEPSHORTCUT_TYPE_KEY).value(true)
|
.key(USER_HANDLE_KEY).value(UserManagerCompat.getInstance(mContext)
|
.getSerialNumberForUser(user))
|
.endObject().toString();
|
} else if (providerInfo != null) {
|
// If it a launcher target, we only need component name, and user to
|
// recreate this.
|
return new JSONStringer()
|
.object()
|
.key(LAUNCH_INTENT_KEY).value(launchIntent.toUri(0))
|
.key(APP_WIDGET_TYPE_KEY).value(true)
|
.key(USER_HANDLE_KEY).value(UserManagerCompat.getInstance(mContext)
|
.getSerialNumberForUser(user))
|
.endObject().toString();
|
}
|
|
if (launchIntent.getAction() == null) {
|
launchIntent.setAction(Intent.ACTION_VIEW);
|
} else if (launchIntent.getAction().equals(Intent.ACTION_MAIN) &&
|
launchIntent.getCategories() != null &&
|
launchIntent.getCategories().contains(Intent.CATEGORY_LAUNCHER)) {
|
launchIntent.addFlags(
|
Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
|
}
|
|
// This name is only used for comparisons and notifications, so fall back to activity
|
// name if not supplied
|
String name = ensureValidName(mContext, launchIntent, label).toString();
|
Bitmap icon = data.getParcelableExtra(Intent.EXTRA_SHORTCUT_ICON);
|
Intent.ShortcutIconResource iconResource =
|
data.getParcelableExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE);
|
|
// Only encode the parameters which are supported by the API.
|
JSONStringer json = new JSONStringer()
|
.object()
|
.key(LAUNCH_INTENT_KEY).value(launchIntent.toUri(0))
|
.key(NAME_KEY).value(name);
|
if (icon != null) {
|
byte[] iconByteArray = GraphicsUtils.flattenBitmap(icon);
|
json = json.key(ICON_KEY).value(
|
Base64.encodeToString(
|
iconByteArray, 0, iconByteArray.length, Base64.DEFAULT));
|
}
|
if (iconResource != null) {
|
json = json.key(ICON_RESOURCE_NAME_KEY).value(iconResource.resourceName);
|
json = json.key(ICON_RESOURCE_PACKAGE_NAME_KEY)
|
.value(iconResource.packageName);
|
}
|
return json.endObject().toString();
|
} catch (JSONException e) {
|
Log.d(TAG, "Exception when adding shortcut: " + e);
|
return null;
|
}
|
}
|
|
public Pair<ItemInfo, Object> getItemInfo() {
|
if (activityInfo != null) {
|
AppInfo appInfo = new AppInfo(mContext, activityInfo, user);
|
final LauncherAppState app = LauncherAppState.getInstance(mContext);
|
// Set default values until proper values is loaded.
|
appInfo.title = "";
|
appInfo.applyFrom(app.getIconCache().getDefaultIcon(user));
|
final WorkspaceItemInfo si = appInfo.makeWorkspaceItem();
|
if (Looper.myLooper() == LauncherModel.getWorkerLooper()) {
|
app.getIconCache().getTitleAndIcon(si, activityInfo, false /* useLowResIcon */);
|
} else {
|
app.getModel().updateAndBindWorkspaceItem(() -> {
|
app.getIconCache().getTitleAndIcon(
|
si, activityInfo, false /* useLowResIcon */);
|
return si;
|
});
|
}
|
return Pair.create((ItemInfo) si, (Object) activityInfo);
|
} else if (shortcutInfo != null) {
|
WorkspaceItemInfo si = new WorkspaceItemInfo(shortcutInfo, mContext);
|
LauncherIcons li = LauncherIcons.obtain(mContext);
|
si.applyFrom(li.createShortcutIcon(shortcutInfo));
|
li.recycle();
|
return Pair.create((ItemInfo) si, (Object) shortcutInfo);
|
} else if (providerInfo != null) {
|
LauncherAppWidgetProviderInfo info = LauncherAppWidgetProviderInfo
|
.fromProviderInfo(mContext, providerInfo);
|
LauncherAppWidgetInfo widgetInfo = new LauncherAppWidgetInfo(
|
launchIntent.getIntExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, 0),
|
info.provider);
|
InvariantDeviceProfile idp = LauncherAppState.getIDP(mContext);
|
widgetInfo.minSpanX = info.minSpanX;
|
widgetInfo.minSpanY = info.minSpanY;
|
widgetInfo.spanX = Math.min(info.spanX, idp.numColumns);
|
widgetInfo.spanY = Math.min(info.spanY, idp.numRows);
|
return Pair.create((ItemInfo) widgetInfo, (Object) providerInfo);
|
} else {
|
WorkspaceItemInfo si = createWorkspaceItemInfo(data, LauncherAppState.getInstance(mContext));
|
return Pair.create((ItemInfo) si, null);
|
}
|
}
|
|
public boolean isLauncherActivity() {
|
return activityInfo != null;
|
}
|
}
|
|
private static String getIntentPackage(Intent intent) {
|
return intent.getComponent() == null
|
? intent.getPackage() : intent.getComponent().getPackageName();
|
}
|
|
private static PendingInstallShortcutInfo decode(String encoded, Context context) {
|
try {
|
Decoder decoder = new Decoder(encoded, context);
|
if (decoder.optBoolean(APP_SHORTCUT_TYPE_KEY)) {
|
LauncherActivityInfo info = LauncherAppsCompat.getInstance(context)
|
.resolveActivity(decoder.launcherIntent, decoder.user);
|
return info == null ? null : new PendingInstallShortcutInfo(info, context);
|
} else if (decoder.optBoolean(DEEPSHORTCUT_TYPE_KEY)) {
|
DeepShortcutManager sm = DeepShortcutManager.getInstance(context);
|
List<ShortcutInfo> si = sm.queryForFullDetails(
|
decoder.launcherIntent.getPackage(),
|
Arrays.asList(decoder.launcherIntent.getStringExtra(
|
ShortcutKey.EXTRA_SHORTCUT_ID)),
|
decoder.user);
|
if (si.isEmpty()) {
|
return null;
|
} else {
|
return new PendingInstallShortcutInfo(si.get(0), context);
|
}
|
} else if (decoder.optBoolean(APP_WIDGET_TYPE_KEY)) {
|
int widgetId = decoder.launcherIntent
|
.getIntExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, 0);
|
AppWidgetProviderInfo info = AppWidgetManager.getInstance(context)
|
.getAppWidgetInfo(widgetId);
|
if (info == null || !info.provider.equals(decoder.launcherIntent.getComponent()) ||
|
!info.getProfile().equals(decoder.user)) {
|
return null;
|
}
|
return new PendingInstallShortcutInfo(info, widgetId, context);
|
}
|
|
Intent data = new Intent();
|
data.putExtra(Intent.EXTRA_SHORTCUT_INTENT, decoder.launcherIntent);
|
data.putExtra(Intent.EXTRA_SHORTCUT_NAME, decoder.getString(NAME_KEY));
|
|
String iconBase64 = decoder.optString(ICON_KEY);
|
String iconResourceName = decoder.optString(ICON_RESOURCE_NAME_KEY);
|
String iconResourcePackageName = decoder.optString(ICON_RESOURCE_PACKAGE_NAME_KEY);
|
if (iconBase64 != null && !iconBase64.isEmpty()) {
|
byte[] iconArray = Base64.decode(iconBase64, Base64.DEFAULT);
|
Bitmap b = BitmapFactory.decodeByteArray(iconArray, 0, iconArray.length);
|
data.putExtra(Intent.EXTRA_SHORTCUT_ICON, b);
|
} else if (iconResourceName != null && !iconResourceName.isEmpty()) {
|
Intent.ShortcutIconResource iconResource =
|
new Intent.ShortcutIconResource();
|
iconResource.resourceName = iconResourceName;
|
iconResource.packageName = iconResourcePackageName;
|
data.putExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE, iconResource);
|
}
|
|
return new PendingInstallShortcutInfo(data, decoder.user, context);
|
} catch (JSONException | URISyntaxException e) {
|
Log.d(TAG, "Exception reading shortcut to add: " + e);
|
}
|
return null;
|
}
|
|
private static class Decoder extends JSONObject {
|
public final Intent launcherIntent;
|
public final UserHandle user;
|
|
private Decoder(String encoded, Context context) throws JSONException, URISyntaxException {
|
super(encoded);
|
launcherIntent = Intent.parseUri(getString(LAUNCH_INTENT_KEY), 0);
|
user = has(USER_HANDLE_KEY) ? UserManagerCompat.getInstance(context)
|
.getUserForSerialNumber(getLong(USER_HANDLE_KEY))
|
: Process.myUserHandle();
|
if (user == null) {
|
throw new JSONException("Invalid user");
|
}
|
}
|
}
|
|
/**
|
* Tries to create a new PendingInstallShortcutInfo which represents the same target,
|
* but is an app target and not a shortcut.
|
* @return the newly created info or the original one.
|
*/
|
private static PendingInstallShortcutInfo convertToLauncherActivityIfPossible(
|
PendingInstallShortcutInfo original) {
|
if (original.isLauncherActivity()) {
|
// Already an activity target
|
return original;
|
}
|
if (!Utilities.isLauncherAppTarget(original.launchIntent)) {
|
return original;
|
}
|
|
LauncherActivityInfo info = LauncherAppsCompat.getInstance(original.mContext)
|
.resolveActivity(original.launchIntent, original.user);
|
if (info == null) {
|
return original;
|
}
|
// Ignore any conflicts in the label name, as that can change based on locale.
|
return new PendingInstallShortcutInfo(info, original.mContext);
|
}
|
|
private static WorkspaceItemInfo createWorkspaceItemInfo(Intent data, LauncherAppState app) {
|
Intent intent = data.getParcelableExtra(Intent.EXTRA_SHORTCUT_INTENT);
|
String name = data.getStringExtra(Intent.EXTRA_SHORTCUT_NAME);
|
Parcelable bitmap = data.getParcelableExtra(Intent.EXTRA_SHORTCUT_ICON);
|
|
if (intent == null) {
|
// If the intent is null, return null as we can't construct a valid WorkspaceItemInfo
|
Log.e(TAG, "Can't construct WorkspaceItemInfo with null intent");
|
return null;
|
}
|
|
final WorkspaceItemInfo info = new WorkspaceItemInfo();
|
|
// Only support intents for current user for now. Intents sent from other
|
// users wouldn't get here without intent forwarding anyway.
|
info.user = Process.myUserHandle();
|
|
BitmapInfo iconInfo = null;
|
LauncherIcons li = LauncherIcons.obtain(app.getContext());
|
if (bitmap instanceof Bitmap) {
|
iconInfo = li.createIconBitmap((Bitmap) bitmap);
|
} else {
|
Parcelable extra = data.getParcelableExtra(Intent.EXTRA_SHORTCUT_ICON_RESOURCE);
|
if (extra instanceof Intent.ShortcutIconResource) {
|
info.iconResource = (Intent.ShortcutIconResource) extra;
|
iconInfo = li.createIconBitmap(info.iconResource);
|
}
|
}
|
li.recycle();
|
|
if (iconInfo == null) {
|
iconInfo = app.getIconCache().getDefaultIcon(info.user);
|
}
|
info.applyFrom(iconInfo);
|
|
info.title = Utilities.trim(name);
|
info.contentDescription = app.getContext().getPackageManager()
|
.getUserBadgedLabel(info.title, info.user);
|
info.intent = intent;
|
return info;
|
}
|
|
}
|