/* * Copyright (C) 2014 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 static com.android.launcher3.LauncherSettings.Favorites.ITEM_TYPE_APPLICATION; import static com.android.launcher3.LauncherSettings.Favorites.TABLE_NAME; import static com.android.launcher3.provider.LauncherDbUtils.itemIdMatch; import static com.android.launcher3.util.UserIconInfo.TYPE_CLONED; import static com.android.launcher3.util.UserIconInfo.TYPE_WORK; import android.content.ComponentName; import android.content.ContentValues; import android.content.Context; import android.content.Intent; import android.content.pm.ActivityInfo; import android.content.pm.LauncherActivityInfo; import android.content.pm.LauncherApps; import android.content.pm.PackageManager; import android.content.res.Resources; import android.content.res.Resources.NotFoundException; import android.content.res.XmlResourceParser; import android.database.sqlite.SQLiteDatabase; import android.os.Bundle; import android.os.Process; import android.os.UserHandle; import android.text.TextUtils; import android.util.ArrayMap; import android.util.AttributeSet; import android.util.Log; import android.util.Xml; import androidx.annotation.Nullable; import androidx.annotation.StringRes; import androidx.annotation.WorkerThread; import androidx.annotation.XmlRes; import com.android.launcher3.LauncherSettings.Favorites; import com.android.launcher3.model.data.AppInfo; import com.android.launcher3.model.data.LauncherAppWidgetInfo; import com.android.launcher3.model.data.WorkspaceItemInfo; import com.android.launcher3.pm.UserCache; import com.android.launcher3.qsb.QsbContainerView; import com.android.launcher3.shortcuts.ShortcutKey; import com.android.launcher3.util.ApiWrapper; import com.android.launcher3.util.IntArray; import com.android.launcher3.util.Partner; import com.android.launcher3.util.Thunk; import com.android.launcher3.util.UserIconInfo; import com.android.launcher3.widget.LauncherWidgetHolder; import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlPullParserException; import java.io.IOException; import java.util.Collections; import java.util.HashMap; import java.util.Locale; import java.util.Map; import java.util.function.Supplier; /** * Layout parsing code for auto installs layout */ public class AutoInstallsLayout { private static final String TAG = "AutoInstalls"; private static final boolean LOGD = false; /** Marker action used to discover a package which defines launcher customization */ static final String ACTION_LAUNCHER_CUSTOMIZATION = "android.autoinstalls.config.action.PLAY_AUTO_INSTALL"; /** * Layout resource which also includes grid size and hotseat count, e.g., default_layout_6x6_h5 */ private static final String FORMATTED_LAYOUT_RES_WITH_HOSTEAT = "default_layout_%dx%d_h%s"; private static final String FORMATTED_LAYOUT_RES = "default_layout_%dx%d"; private static final String LAYOUT_RES = "default_layout"; public static AutoInstallsLayout get(Context context, LauncherWidgetHolder appWidgetHolder, LayoutParserCallback callback) { Partner partner = Partner.get(context.getPackageManager(), ACTION_LAUNCHER_CUSTOMIZATION); if (partner == null) { return null; } InvariantDeviceProfile grid = LauncherAppState.getIDP(context); // Try with grid size and hotseat count String layoutName = String.format(Locale.ENGLISH, FORMATTED_LAYOUT_RES_WITH_HOSTEAT, grid.numColumns, grid.numRows, grid.numDatabaseHotseatIcons); int layoutId = partner.getXmlResId(layoutName); // Try with only grid size if (layoutId == 0) { Log.d(TAG, "Formatted layout: " + layoutName + " not found. Trying layout without hosteat"); layoutName = String.format(Locale.ENGLISH, FORMATTED_LAYOUT_RES, grid.numColumns, grid.numRows); layoutId = partner.getXmlResId(layoutName); } // Try the default layout if (layoutId == 0) { Log.d(TAG, "Formatted layout: " + layoutName + " not found. Trying the default layout"); layoutId = partner.getXmlResId(LAYOUT_RES); } if (layoutId == 0) { Log.e(TAG, "Layout definition not found in package: " + partner.getPackageName()); return null; } return new AutoInstallsLayout(context, appWidgetHolder, callback, partner.getResources(), layoutId, TAG_WORKSPACE); } // Object Tags private static final String TAG_INCLUDE = "include"; public static final String TAG_WORKSPACE = "workspace"; private static final String TAG_APP_ICON = "appicon"; public static final String TAG_AUTO_INSTALL = "autoinstall"; public static final String TAG_FOLDER = "folder"; public static final String TAG_APPWIDGET = "appwidget"; protected static final String TAG_SEARCH_WIDGET = "searchwidget"; public static final String TAG_SHORTCUT = "shortcut"; private static final String TAG_EXTRA = "extra"; public static final String ATTR_CONTAINER = "container"; public static final String ATTR_RANK = "rank"; public static final String ATTR_PACKAGE_NAME = "packageName"; public static final String ATTR_CLASS_NAME = "className"; public static final String ATTR_TITLE = "title"; public static final String ATTR_TITLE_TEXT = "titleText"; public static final String ATTR_SCREEN = "screen"; public static final String ATTR_SHORTCUT_ID = "shortcutId"; // x and y can be specified as negative integers, in which case -1 represents the // last row / column, -2 represents the second last, and so on. public static final String ATTR_X = "x"; public static final String ATTR_Y = "y"; public static final String ATTR_SPAN_X = "spanX"; public static final String ATTR_SPAN_Y = "spanY"; // Attrs for "Include" private static final String ATTR_WORKSPACE = "workspace"; public static final String ATTR_USER_TYPE = "userType"; public static final String USER_TYPE_WORK = "work"; public static final String USER_TYPE_CLONED = "cloned"; // Style attrs -- "Extra" private static final String ATTR_KEY = "key"; private static final String ATTR_VALUE = "value"; private static final String HOTSEAT_CONTAINER_NAME = Favorites.containerToString(Favorites.CONTAINER_HOTSEAT); protected final Context mContext; protected final LauncherWidgetHolder mAppWidgetHolder; protected final LayoutParserCallback mCallback; protected final PackageManager mPackageManager; protected final SourceResources mSourceRes; protected final Supplier mInitialLayoutSupplier; private final Map mUserTypeToSerial; private final InvariantDeviceProfile mIdp; private final int mRowCount; private final int mColumnCount; private final Map mActivityOverride; private final int[] mTemp = new int[2]; @Thunk final ContentValues mValues; protected final String mRootTag; protected SQLiteDatabase mDb; public AutoInstallsLayout(Context context, LauncherWidgetHolder appWidgetHolder, LayoutParserCallback callback, Resources res, int layoutId, String rootTag) { this(context, appWidgetHolder, callback, SourceResources.wrap(res), () -> res.getXml(layoutId), rootTag); } public AutoInstallsLayout(Context context, LauncherWidgetHolder appWidgetHolder, LayoutParserCallback callback, SourceResources res, Supplier initialLayoutSupplier, String rootTag) { mContext = context; mAppWidgetHolder = appWidgetHolder; mCallback = callback; mPackageManager = context.getPackageManager(); mValues = new ContentValues(); mRootTag = rootTag; mSourceRes = res; mInitialLayoutSupplier = initialLayoutSupplier; mIdp = LauncherAppState.getIDP(context); mRowCount = mIdp.numRows; mColumnCount = mIdp.numColumns; mActivityOverride = ApiWrapper.INSTANCE.get(context).getActivityOverrides(); mUserTypeToSerial = new HashMap<>(); UserCache cache = UserCache.getInstance(context); for (UserHandle user : cache.getUserProfiles()) { UserIconInfo uii = cache.getUserInfo(user); switch (uii.type) { case TYPE_WORK -> mUserTypeToSerial.put(USER_TYPE_WORK, uii.userSerial); case TYPE_CLONED -> mUserTypeToSerial.put(USER_TYPE_CLONED, uii.userSerial); } } } /** * Loads the layout in the db and returns the number of entries added on the desktop. */ public int loadLayout(SQLiteDatabase db) { mDb = db; try { return parseLayout(mInitialLayoutSupplier.get()); } catch (Exception e) { Log.e(TAG, "Error parsing layout: ", e); return -1; } } /** * Parses the layout and returns the number of elements added on the homescreen. */ protected int parseLayout(XmlPullParser parser) throws XmlPullParserException, IOException { beginDocument(parser, mRootTag); final int depth = parser.getDepth(); int type; ArrayMap tagParserMap = getLayoutElementsMap(); int count = 0; while (((type = parser.next()) != XmlPullParser.END_TAG || parser.getDepth() > depth) && type != XmlPullParser.END_DOCUMENT) { if (type != XmlPullParser.START_TAG) { continue; } count += parseAndAddNode(parser, tagParserMap); } return count; } /** * Parses container and screenId attribute from the current tag, and puts it in the out. * @param out array of size 2. */ protected void parseContainerAndScreen(XmlPullParser parser, int[] out) { if (HOTSEAT_CONTAINER_NAME.equals(getAttributeValue(parser, ATTR_CONTAINER))) { out[0] = Favorites.CONTAINER_HOTSEAT; // Hack: hotseat items are stored using screen ids out[1] = Integer.parseInt(getAttributeValue(parser, ATTR_RANK)); } else { out[0] = Favorites.CONTAINER_DESKTOP; out[1] = Integer.parseInt(getAttributeValue(parser, ATTR_SCREEN)); } } /** * Parses the current node and returns the number of elements added. */ protected int parseAndAddNode( XmlPullParser parser, ArrayMap tagParserMap) throws XmlPullParserException, IOException { if (TAG_INCLUDE.equals(parser.getName())) { final int resId = getAttributeResourceValue(parser, ATTR_WORKSPACE, 0); if (resId != 0) { // recursively load some more favorites, why not? return parseLayout(mSourceRes.getXml(resId)); } else { return 0; } } mValues.clear(); parseContainerAndScreen(parser, mTemp); final int container = mTemp[0]; final int screenId = mTemp[1]; mValues.put(Favorites.CONTAINER, container); mValues.put(Favorites.SCREEN, screenId); mValues.put(Favorites.CELLX, convertToDistanceFromEnd(getAttributeValue(parser, ATTR_X), mColumnCount)); mValues.put(Favorites.CELLY, convertToDistanceFromEnd(getAttributeValue(parser, ATTR_Y), mRowCount)); Long profileId = mUserTypeToSerial.get(getAttributeValue(parser, ATTR_USER_TYPE)); if (profileId != null) { mValues.put(Favorites.PROFILE_ID, profileId); } TagParser tagParser = tagParserMap.get(parser.getName()); if (tagParser == null) { if (LOGD) Log.d(TAG, "Ignoring unknown element tag: " + parser.getName()); return 0; } return tagParser.parseAndAdd(parser) >= 0 ? 1 : 0; } protected int addShortcut(String title, Intent intent, int type) { int id = mCallback.generateNewItemId(); mValues.put(Favorites.INTENT, intent.toUri(0)); mValues.put(Favorites.TITLE, title); mValues.put(Favorites.ITEM_TYPE, type); mValues.put(Favorites.SPANX, 1); mValues.put(Favorites.SPANY, 1); mValues.put(Favorites._ID, id); ComponentName cn = intent.getComponent(); if (cn != null && type == ITEM_TYPE_APPLICATION && !mValues.containsKey(Favorites.PROFILE_ID)) { LauncherActivityInfo replacementInfo = mActivityOverride.get(cn.getPackageName()); if (replacementInfo != null) { mValues.put(Favorites.PROFILE_ID, UserCache.INSTANCE.get(mContext) .getSerialNumberForUser(replacementInfo.getUser())); mValues.put(Favorites.INTENT, AppInfo.makeLaunchIntent(replacementInfo).toUri(0)); } } if (mCallback.insertAndCheck(mDb, mValues) < 0) { return -1; } else { return id; } } protected ArrayMap getFolderElementsMap() { ArrayMap parsers = new ArrayMap<>(); parsers.put(TAG_APP_ICON, new AppShortcutParser()); parsers.put(TAG_AUTO_INSTALL, new AutoInstallParser()); parsers.put(TAG_SHORTCUT, new ShortcutParser()); return parsers; } protected ArrayMap getLayoutElementsMap() { ArrayMap parsers = new ArrayMap<>(); parsers.put(TAG_APP_ICON, new AppShortcutParser()); parsers.put(TAG_AUTO_INSTALL, new AutoInstallParser()); parsers.put(TAG_FOLDER, new FolderParser()); parsers.put(TAG_APPWIDGET, new PendingWidgetParser()); parsers.put(TAG_SEARCH_WIDGET, new SearchWidgetParser()); parsers.put(TAG_SHORTCUT, new ShortcutParser()); return parsers; } protected interface TagParser { /** * Parses the tag and adds to the db * @return the id of the row added or -1; */ int parseAndAdd(XmlPullParser parser) throws XmlPullParserException, IOException; } /** * App shortcuts: required attributes packageName and className */ protected class AppShortcutParser implements TagParser { @Override public int parseAndAdd(XmlPullParser parser) { final String packageName = getAttributeValue(parser, ATTR_PACKAGE_NAME); final String className = getAttributeValue(parser, ATTR_CLASS_NAME); if (!TextUtils.isEmpty(packageName) && !TextUtils.isEmpty(className)) { ActivityInfo info; try { ComponentName cn; try { cn = new ComponentName(packageName, className); info = mPackageManager.getActivityInfo(cn, 0); } catch (PackageManager.NameNotFoundException nnfe) { String[] packages = mPackageManager.currentToCanonicalPackageNames( new String[]{packageName}); cn = new ComponentName(packages[0], className); info = mPackageManager.getActivityInfo(cn, 0); } final Intent intent = new Intent(Intent.ACTION_MAIN, null) .addCategory(Intent.CATEGORY_LAUNCHER) .setComponent(cn) .setFlags(Intent.FLAG_ACTIVITY_NEW_TASK | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED); return addShortcut(info.loadLabel(mPackageManager).toString(), intent, ITEM_TYPE_APPLICATION); } catch (PackageManager.NameNotFoundException e) { Log.e(TAG, "Favorite not found: " + packageName + "/" + className); } return -1; } else { return invalidPackageOrClass(parser); } } /** * Helper method to allow extending the parser capabilities */ protected int invalidPackageOrClass(XmlPullParser parser) { Log.w(TAG, "Skipping invalid with no component"); return -1; } } /** * AutoInstall: required attributes packageName and className */ protected class AutoInstallParser implements TagParser { @Override public int parseAndAdd(XmlPullParser parser) { final String packageName = getAttributeValue(parser, ATTR_PACKAGE_NAME); final String className = getAttributeValue(parser, ATTR_CLASS_NAME); if (TextUtils.isEmpty(packageName) || TextUtils.isEmpty(className)) { if (LOGD) Log.d(TAG, "Skipping invalid with no component"); return -1; } mValues.put(Favorites.RESTORED, WorkspaceItemInfo.FLAG_AUTOINSTALL_ICON); Intent intent = AppInfo.makeLaunchIntent(new ComponentName(packageName, className)); return addShortcut(mContext.getString(R.string.package_state_unknown), intent, ITEM_TYPE_APPLICATION); } } /** * Parses a deep shortcut. Required attributes packageName and shortcutId */ protected class ShortcutParser implements TagParser { @Override public int parseAndAdd(XmlPullParser parser) { final String packageName = getAttributeValue(parser, ATTR_PACKAGE_NAME); final String shortcutId = getAttributeValue(parser, ATTR_SHORTCUT_ID); try { LauncherApps launcherApps = mContext.getSystemService(LauncherApps.class); launcherApps.pinShortcuts(packageName, Collections.singletonList(shortcutId), Process.myUserHandle()); Intent intent = ShortcutKey.makeIntent(shortcutId, packageName); mValues.put(Favorites.RESTORED, WorkspaceItemInfo.FLAG_RESTORED_ICON); return addShortcut(null, intent, Favorites.ITEM_TYPE_DEEP_SHORTCUT); } catch (Exception e) { Log.e(TAG, "Unable to pin the shortcut for shortcut id = " + shortcutId + " and package name = " + packageName, e); } return -1; } } /** * AppWidget parser: Required attributes packageName, className, spanX and spanY. * Options child nodes: * It adds a pending widget which allows the widget to come later. If there are extras, those * are passed to widget options during bind. * The config activity for the widget (if present) is not shown, so any optional configurations * should be passed as extras and the widget should support reading these widget options. */ protected class PendingWidgetParser implements TagParser { @Nullable public ComponentName getComponentName(XmlPullParser parser) { final String packageName = getAttributeValue(parser, ATTR_PACKAGE_NAME); final String className = getAttributeValue(parser, ATTR_CLASS_NAME); if (TextUtils.isEmpty(packageName) || TextUtils.isEmpty(className)) { return null; } return new ComponentName(packageName, className); } @Override public int parseAndAdd(XmlPullParser parser) throws XmlPullParserException, IOException { ComponentName cn = getComponentName(parser); if (cn == null) { if (LOGD) Log.d(TAG, "Skipping invalid with no component"); return -1; } mValues.put(Favorites.SPANX, getAttributeValue(parser, ATTR_SPAN_X)); mValues.put(Favorites.SPANY, getAttributeValue(parser, ATTR_SPAN_Y)); mValues.put(Favorites.ITEM_TYPE, Favorites.ITEM_TYPE_APPWIDGET); // Read the extras Bundle extras = new Bundle(); int widgetDepth = parser.getDepth(); int type; while ((type = parser.next()) != XmlPullParser.END_TAG || parser.getDepth() > widgetDepth) { if (type != XmlPullParser.START_TAG) { continue; } if (TAG_EXTRA.equals(parser.getName())) { String key = getAttributeValue(parser, ATTR_KEY); String value = getAttributeValue(parser, ATTR_VALUE); if (key != null && value != null) { extras.putString(key, value); } else { throw new RuntimeException("Widget extras must have a key and value"); } } else { throw new RuntimeException("Widgets can contain only extras"); } } return verifyAndInsert(cn, extras); } protected int verifyAndInsert(ComponentName cn, Bundle extras) { mValues.put(Favorites.APPWIDGET_PROVIDER, cn.flattenToString()); mValues.put(Favorites.RESTORED, LauncherAppWidgetInfo.FLAG_ID_NOT_VALID | LauncherAppWidgetInfo.FLAG_PROVIDER_NOT_READY | LauncherAppWidgetInfo.FLAG_DIRECT_CONFIG); mValues.put(Favorites._ID, mCallback.generateNewItemId()); if (!extras.isEmpty()) { mValues.put(Favorites.INTENT, new Intent().putExtras(extras).toUri(0)); } int insertedId = mCallback.insertAndCheck(mDb, mValues); if (insertedId < 0) { return -1; } else { return insertedId; } } } protected class SearchWidgetParser extends PendingWidgetParser { @Override @Nullable @WorkerThread public ComponentName getComponentName(XmlPullParser parser) { return QsbContainerView.getSearchComponentName(mContext); } @Override protected int verifyAndInsert(ComponentName cn, Bundle extras) { mValues.put(Favorites.OPTIONS, LauncherAppWidgetInfo.OPTION_SEARCH_WIDGET); int flags = mValues.getAsInteger(Favorites.RESTORED) | WorkspaceItemInfo.FLAG_RESTORE_STARTED; mValues.put(Favorites.RESTORED, flags); return super.verifyAndInsert(cn, extras); } } protected class FolderParser implements TagParser { private final ArrayMap mFolderElements; public FolderParser() { this(getFolderElementsMap()); } public FolderParser(ArrayMap elements) { mFolderElements = elements; } @Override public int parseAndAdd(XmlPullParser parser) throws XmlPullParserException, IOException { final String title; final int titleResId = getAttributeResourceValue(parser, ATTR_TITLE, 0); if (titleResId != 0) { title = mSourceRes.getString(titleResId); } else { String titleText = getAttributeValue(parser, ATTR_TITLE_TEXT); title = TextUtils.isEmpty(titleText) ? "" : titleText; } mValues.put(Favorites.TITLE, title); mValues.put(Favorites.ITEM_TYPE, Favorites.ITEM_TYPE_FOLDER); mValues.put(Favorites.SPANX, 1); mValues.put(Favorites.SPANY, 1); mValues.put(Favorites._ID, mCallback.generateNewItemId()); int folderId = mCallback.insertAndCheck(mDb, mValues); if (folderId < 0) { if (LOGD) Log.e(TAG, "Unable to add folder"); return -1; } final ContentValues myValues = new ContentValues(mValues); IntArray folderItems = new IntArray(); int type; int folderDepth = parser.getDepth(); int rank = 0; while ((type = parser.next()) != XmlPullParser.END_TAG || parser.getDepth() > folderDepth) { if (type != XmlPullParser.START_TAG) { continue; } mValues.clear(); mValues.put(Favorites.CONTAINER, folderId); mValues.put(Favorites.RANK, rank); TagParser tagParser = mFolderElements.get(parser.getName()); if (tagParser != null) { final int id = tagParser.parseAndAdd(parser); if (id >= 0) { folderItems.add(id); rank++; } } else { throw new RuntimeException("Invalid folder item " + parser.getName()); } } int addedId = folderId; // We can only have folders with >= 2 items, so we need to remove the // folder and clean up if less than 2 items were included, or some // failed to add, and less than 2 were actually added if (folderItems.size() < 2) { // Delete the folder mDb.delete(TABLE_NAME, itemIdMatch(folderId), null); addedId = -1; // If we have a single item, promote it to where the folder // would have been. if (folderItems.size() == 1) { final ContentValues childValues = new ContentValues(); copyInteger(myValues, childValues, Favorites.CONTAINER); copyInteger(myValues, childValues, Favorites.SCREEN); copyInteger(myValues, childValues, Favorites.CELLX); copyInteger(myValues, childValues, Favorites.CELLY); addedId = folderItems.get(0); mDb.update(TABLE_NAME, childValues, Favorites._ID + "=" + addedId, null); } } return addedId; } } public static void beginDocument(XmlPullParser parser, String firstElementName) throws XmlPullParserException, IOException { int type; while ((type = parser.next()) != XmlPullParser.START_TAG && type != XmlPullParser.END_DOCUMENT); if (type != XmlPullParser.START_TAG) { throw new XmlPullParserException("No start tag found"); } if (!parser.getName().equals(firstElementName)) { throw new XmlPullParserException("Unexpected start tag: found " + parser.getName() + ", expected " + firstElementName); } } private static String convertToDistanceFromEnd(String value, int endValue) { if (!TextUtils.isEmpty(value)) { int x = Integer.parseInt(value); if (x < 0) { return Integer.toString(endValue + x); } } return value; } /** * Return attribute value, attempting launcher-specific namespace first * before falling back to anonymous attribute. */ protected static String getAttributeValue(XmlPullParser parser, String attribute) { String value = parser.getAttributeValue( "http://schemas.android.com/apk/res-auto/com.android.launcher3", attribute); if (value == null) { value = parser.getAttributeValue(null, attribute); } return value; } /** * Return attribute resource value, attempting launcher-specific namespace * first before falling back to anonymous attribute. */ protected static int getAttributeResourceValue(XmlPullParser parser, String attribute, int defaultValue) { AttributeSet attrs = Xml.asAttributeSet(parser); int value = attrs.getAttributeResourceValue( "http://schemas.android.com/apk/res-auto/com.android.launcher3", attribute, defaultValue); if (value == defaultValue) { value = attrs.getAttributeResourceValue(null, attribute, defaultValue); } return value; } public interface LayoutParserCallback { int generateNewItemId(); int insertAndCheck(SQLiteDatabase db, ContentValues values); } @Thunk static void copyInteger(ContentValues from, ContentValues to, String key) { to.put(key, from.getAsInteger(key)); } /** * Wrapper over resources for easier abstraction */ public interface SourceResources { /** * Refer {@link Resources#getXml(int)} */ default XmlResourceParser getXml(@XmlRes int id) throws NotFoundException { throw new NotFoundException(); } /** * Refer {@link Resources#getString(int)} */ default String getString(@StringRes int id) throws NotFoundException { throw new NotFoundException(); } /** * Returns a {@link SourceResources} corresponding to the provided resources */ static SourceResources wrap(Resources res) { return new SourceResources() { @Override public XmlResourceParser getXml(int id) { return res.getXml(id); } @Override public String getString(int id) { return res.getString(id); } }; } } }