/*
|
* 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 android.appwidget.AppWidgetHost;
|
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.PackageManager;
|
import android.content.res.Resources;
|
import android.database.sqlite.SQLiteDatabase;
|
import android.graphics.drawable.Drawable;
|
import android.net.Uri;
|
import android.os.Build.VERSION;
|
import android.os.Bundle;
|
import android.os.Process;
|
import android.text.TextUtils;
|
import android.util.ArrayMap;
|
import android.util.AttributeSet;
|
import android.util.Log;
|
import android.util.Pair;
|
import android.util.Patterns;
|
import android.util.Xml;
|
|
import com.android.launcher3.LauncherProvider.SqlArguments;
|
import com.android.launcher3.LauncherSettings.Favorites;
|
import com.android.launcher3.icons.GraphicsUtils;
|
import com.android.launcher3.icons.LauncherIcons;
|
import com.android.launcher3.util.IntArray;
|
import com.android.launcher3.util.Thunk;
|
|
import org.xmlpull.v1.XmlPullParser;
|
import org.xmlpull.v1.XmlPullParserException;
|
|
import java.io.IOException;
|
import java.util.Locale;
|
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";
|
|
static AutoInstallsLayout get(Context context, AppWidgetHost appWidgetHost,
|
LayoutParserCallback callback) {
|
Pair<String, Resources> customizationApkInfo = Utilities.findSystemApk(
|
ACTION_LAUNCHER_CUSTOMIZATION, context.getPackageManager());
|
if (customizationApkInfo == null) {
|
return null;
|
}
|
String pkg = customizationApkInfo.first;
|
Resources targetRes = customizationApkInfo.second;
|
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.numHotseatIcons);
|
int layoutId = targetRes.getIdentifier(layoutName, "xml", pkg);
|
|
// 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 = targetRes.getIdentifier(layoutName, "xml", pkg);
|
}
|
|
// Try the default layout
|
if (layoutId == 0) {
|
Log.d(TAG, "Formatted layout: " + layoutName + " not found. Trying the default layout");
|
layoutId = targetRes.getIdentifier(LAYOUT_RES, "xml", pkg);
|
}
|
|
if (layoutId == 0) {
|
Log.e(TAG, "Layout definition not found in package: " + pkg);
|
return null;
|
}
|
return new AutoInstallsLayout(context, appWidgetHost, callback, targetRes, 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";
|
private static final String TAG_AUTO_INSTALL = "autoinstall";
|
private static final String TAG_FOLDER = "folder";
|
private static final String TAG_APPWIDGET = "appwidget";
|
private static final String TAG_SHORTCUT = "shortcut";
|
private static final String TAG_EXTRA = "extra";
|
|
private static final String ATTR_CONTAINER = "container";
|
private static final String ATTR_RANK = "rank";
|
|
private static final String ATTR_PACKAGE_NAME = "packageName";
|
private static final String ATTR_CLASS_NAME = "className";
|
private static final String ATTR_TITLE = "title";
|
private static final String ATTR_SCREEN = "screen";
|
|
// 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.
|
private static final String ATTR_X = "x";
|
private static final String ATTR_Y = "y";
|
|
private static final String ATTR_SPAN_X = "spanX";
|
private static final String ATTR_SPAN_Y = "spanY";
|
private static final String ATTR_ICON = "icon";
|
private static final String ATTR_URL = "url";
|
|
// Attrs for "Include"
|
private static final String ATTR_WORKSPACE = "workspace";
|
|
// 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);
|
|
@Thunk final Context mContext;
|
@Thunk final AppWidgetHost mAppWidgetHost;
|
protected final LayoutParserCallback mCallback;
|
|
protected final PackageManager mPackageManager;
|
protected final Resources mSourceRes;
|
protected final Supplier<XmlPullParser> mInitialLayoutSupplier;
|
|
private final InvariantDeviceProfile mIdp;
|
private final int mRowCount;
|
private final int mColumnCount;
|
|
private final int[] mTemp = new int[2];
|
@Thunk final ContentValues mValues;
|
protected final String mRootTag;
|
|
protected SQLiteDatabase mDb;
|
|
public AutoInstallsLayout(Context context, AppWidgetHost appWidgetHost,
|
LayoutParserCallback callback, Resources res,
|
int layoutId, String rootTag) {
|
this(context, appWidgetHost, callback, res, () -> res.getXml(layoutId), rootTag);
|
}
|
|
public AutoInstallsLayout(Context context, AppWidgetHost appWidgetHost,
|
LayoutParserCallback callback, Resources res,
|
Supplier<XmlPullParser> initialLayoutSupplier, String rootTag) {
|
mContext = context;
|
mAppWidgetHost = appWidgetHost;
|
mCallback = callback;
|
|
mPackageManager = context.getPackageManager();
|
mValues = new ContentValues();
|
mRootTag = rootTag;
|
|
mSourceRes = res;
|
mInitialLayoutSupplier = initialLayoutSupplier;
|
|
mIdp = LauncherAppState.getIDP(context);
|
mRowCount = mIdp.numRows;
|
mColumnCount = mIdp.numColumns;
|
}
|
|
/**
|
* Loads the layout in the db and returns the number of entries added on the desktop.
|
*/
|
public int loadLayout(SQLiteDatabase db, IntArray screenIds) {
|
mDb = db;
|
try {
|
return parseLayout(mInitialLayoutSupplier.get(), screenIds);
|
} 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, IntArray screenIds)
|
throws XmlPullParserException, IOException {
|
beginDocument(parser, mRootTag);
|
final int depth = parser.getDepth();
|
int type;
|
ArrayMap<String, TagParser> 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, screenIds);
|
}
|
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<String, TagParser> tagParserMap, IntArray screenIds)
|
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), screenIds);
|
} 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));
|
|
TagParser tagParser = tagParserMap.get(parser.getName());
|
if (tagParser == null) {
|
if (LOGD) Log.d(TAG, "Ignoring unknown element tag: " + parser.getName());
|
return 0;
|
}
|
int newElementId = tagParser.parseAndAdd(parser);
|
if (newElementId >= 0) {
|
// Keep track of the set of screens which need to be added to the db.
|
if (!screenIds.contains(screenId) &&
|
container == Favorites.CONTAINER_DESKTOP) {
|
screenIds.add(screenId);
|
}
|
return 1;
|
}
|
return 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);
|
if (mCallback.insertAndCheck(mDb, mValues) < 0) {
|
return -1;
|
} else {
|
return id;
|
}
|
}
|
|
protected ArrayMap<String, TagParser> getFolderElementsMap() {
|
ArrayMap<String, TagParser> parsers = new ArrayMap<>();
|
parsers.put(TAG_APP_ICON, new AppShortcutParser());
|
parsers.put(TAG_AUTO_INSTALL, new AutoInstallParser());
|
parsers.put(TAG_SHORTCUT, new ShortcutParser(mSourceRes));
|
return parsers;
|
}
|
|
protected ArrayMap<String, TagParser> getLayoutElementsMap() {
|
ArrayMap<String, TagParser> 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_SHORTCUT, new ShortcutParser(mSourceRes));
|
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, Favorites.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 <favorite> 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 <favorite> with no component");
|
return -1;
|
}
|
|
mValues.put(Favorites.RESTORED, WorkspaceItemInfo.FLAG_AUTOINSTALL_ICON);
|
final Intent intent = new Intent(Intent.ACTION_MAIN, null)
|
.addCategory(Intent.CATEGORY_LAUNCHER)
|
.setComponent(new ComponentName(packageName, className))
|
.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK |
|
Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
|
return addShortcut(mContext.getString(R.string.package_state_unknown), intent,
|
Favorites.ITEM_TYPE_APPLICATION);
|
}
|
}
|
|
/**
|
* Parses a web shortcut. Required attributes url, icon, title
|
*/
|
protected class ShortcutParser implements TagParser {
|
|
private final Resources mIconRes;
|
|
public ShortcutParser(Resources iconRes) {
|
mIconRes = iconRes;
|
}
|
|
@Override
|
public int parseAndAdd(XmlPullParser parser) {
|
final int titleResId = getAttributeResourceValue(parser, ATTR_TITLE, 0);
|
final int iconId = getAttributeResourceValue(parser, ATTR_ICON, 0);
|
|
if (titleResId == 0 || iconId == 0) {
|
if (LOGD) Log.d(TAG, "Ignoring shortcut");
|
return -1;
|
}
|
|
final Intent intent = parseIntent(parser);
|
if (intent == null) {
|
return -1;
|
}
|
|
Drawable icon = mIconRes.getDrawable(iconId);
|
if (icon == null) {
|
if (LOGD) Log.d(TAG, "Ignoring shortcut, can't load icon");
|
return -1;
|
}
|
|
// Auto installs should always support the current platform version.
|
LauncherIcons li = LauncherIcons.obtain(mContext);
|
mValues.put(LauncherSettings.Favorites.ICON, GraphicsUtils.flattenBitmap(
|
li.createBadgedIconBitmap(icon, Process.myUserHandle(), VERSION.SDK_INT).icon));
|
li.recycle();
|
|
mValues.put(Favorites.ICON_PACKAGE, mIconRes.getResourcePackageName(iconId));
|
mValues.put(Favorites.ICON_RESOURCE, mIconRes.getResourceName(iconId));
|
|
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK |
|
Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
|
return addShortcut(mSourceRes.getString(titleResId),
|
intent, Favorites.ITEM_TYPE_SHORTCUT);
|
}
|
|
protected Intent parseIntent(XmlPullParser parser) {
|
final String url = getAttributeValue(parser, ATTR_URL);
|
if (TextUtils.isEmpty(url) || !Patterns.WEB_URL.matcher(url).matches()) {
|
if (LOGD) Log.d(TAG, "Ignoring shortcut, invalid url: " + url);
|
return null;
|
}
|
return new Intent(Intent.ACTION_VIEW, null).setData(Uri.parse(url));
|
}
|
}
|
|
/**
|
* AppWidget parser: Required attributes packageName, className, spanX and spanY.
|
* Options child nodes: <extra key=... value=... />
|
* 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 {
|
|
@Override
|
public int parseAndAdd(XmlPullParser parser)
|
throws XmlPullParserException, IOException {
|
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 <appwidget> 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(new ComponentName(packageName, className), 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 FolderParser implements TagParser {
|
private final ArrayMap<String, TagParser> mFolderElements;
|
|
public FolderParser() {
|
this(getFolderElementsMap());
|
}
|
|
public FolderParser(ArrayMap<String, TagParser> 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 {
|
title = mContext.getResources().getString(R.string.folder_name);
|
}
|
|
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
|
Uri uri = Favorites.getContentUri(folderId);
|
SqlArguments args = new SqlArguments(uri, null, null);
|
mDb.delete(args.table, args.where, args.args);
|
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(Favorites.TABLE_NAME, childValues,
|
Favorites._ID + "=" + addedId, null);
|
}
|
}
|
return addedId;
|
}
|
}
|
|
protected 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));
|
}
|
}
|