Settings Injection used to start activities with startActivityForResult() and its extension, where there was not result really being handled. Hence, this CL corrects the way it starts activities by removing the "ForResult" part. Fixes: 197704126 Test: robotest and launch Android Auto to see it being launched in another task. Change-Id: I259b2555cf62f8966afdf664b337af5f216b3843
460 lines
20 KiB
Java
460 lines
20 KiB
Java
/*
|
|
* Copyright (C) 2015 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.settings.dashboard;
|
|
|
|
import static android.content.Intent.EXTRA_USER;
|
|
|
|
import static com.android.settingslib.drawer.SwitchesProvider.EXTRA_SWITCH_CHECKED_STATE;
|
|
import static com.android.settingslib.drawer.SwitchesProvider.EXTRA_SWITCH_SET_CHECKED_ERROR;
|
|
import static com.android.settingslib.drawer.SwitchesProvider.EXTRA_SWITCH_SET_CHECKED_ERROR_MESSAGE;
|
|
import static com.android.settingslib.drawer.SwitchesProvider.METHOD_GET_DYNAMIC_SUMMARY;
|
|
import static com.android.settingslib.drawer.SwitchesProvider.METHOD_GET_DYNAMIC_TITLE;
|
|
import static com.android.settingslib.drawer.SwitchesProvider.METHOD_GET_PROVIDER_ICON;
|
|
import static com.android.settingslib.drawer.SwitchesProvider.METHOD_IS_CHECKED;
|
|
import static com.android.settingslib.drawer.SwitchesProvider.METHOD_ON_CHECKED_CHANGED;
|
|
import static com.android.settingslib.drawer.TileUtils.META_DATA_PREFERENCE_ICON_URI;
|
|
import static com.android.settingslib.drawer.TileUtils.META_DATA_PREFERENCE_SUMMARY;
|
|
import static com.android.settingslib.drawer.TileUtils.META_DATA_PREFERENCE_SUMMARY_URI;
|
|
import static com.android.settingslib.drawer.TileUtils.META_DATA_PREFERENCE_SWITCH_URI;
|
|
import static com.android.settingslib.drawer.TileUtils.META_DATA_PREFERENCE_TITLE;
|
|
import static com.android.settingslib.drawer.TileUtils.META_DATA_PREFERENCE_TITLE_URI;
|
|
|
|
import android.app.settings.SettingsEnums;
|
|
import android.content.ComponentName;
|
|
import android.content.Context;
|
|
import android.content.IContentProvider;
|
|
import android.content.Intent;
|
|
import android.content.pm.PackageManager;
|
|
import android.graphics.drawable.Drawable;
|
|
import android.graphics.drawable.Icon;
|
|
import android.net.Uri;
|
|
import android.os.Bundle;
|
|
import android.os.UserHandle;
|
|
import android.provider.Settings;
|
|
import android.text.TextUtils;
|
|
import android.util.ArrayMap;
|
|
import android.util.Log;
|
|
import android.util.Pair;
|
|
import android.widget.Toast;
|
|
|
|
import androidx.annotation.VisibleForTesting;
|
|
import androidx.fragment.app.FragmentActivity;
|
|
import androidx.preference.Preference;
|
|
import androidx.preference.SwitchPreference;
|
|
|
|
import com.android.settings.R;
|
|
import com.android.settings.SettingsActivity;
|
|
import com.android.settings.Utils;
|
|
import com.android.settings.dashboard.profileselector.ProfileSelectDialog;
|
|
import com.android.settings.overlay.FeatureFactory;
|
|
import com.android.settings.widget.PrimarySwitchPreference;
|
|
import com.android.settingslib.core.instrumentation.MetricsFeatureProvider;
|
|
import com.android.settingslib.drawer.ActivityTile;
|
|
import com.android.settingslib.drawer.CategoryKey;
|
|
import com.android.settingslib.drawer.DashboardCategory;
|
|
import com.android.settingslib.drawer.Tile;
|
|
import com.android.settingslib.drawer.TileUtils;
|
|
import com.android.settingslib.utils.ThreadUtils;
|
|
import com.android.settingslib.widget.AdaptiveIcon;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
import java.util.Map;
|
|
|
|
/**
|
|
* Impl for {@code DashboardFeatureProvider}.
|
|
*/
|
|
public class DashboardFeatureProviderImpl implements DashboardFeatureProvider {
|
|
|
|
private static final String TAG = "DashboardFeatureImpl";
|
|
private static final String DASHBOARD_TILE_PREF_KEY_PREFIX = "dashboard_tile_pref_";
|
|
private static final String META_DATA_KEY_INTENT_ACTION = "com.android.settings.intent.action";
|
|
|
|
protected final Context mContext;
|
|
|
|
private final MetricsFeatureProvider mMetricsFeatureProvider;
|
|
private final CategoryManager mCategoryManager;
|
|
private final PackageManager mPackageManager;
|
|
|
|
public DashboardFeatureProviderImpl(Context context) {
|
|
mContext = context.getApplicationContext();
|
|
mCategoryManager = CategoryManager.get(context);
|
|
mMetricsFeatureProvider = FeatureFactory.getFactory(context).getMetricsFeatureProvider();
|
|
mPackageManager = context.getPackageManager();
|
|
}
|
|
|
|
@Override
|
|
public DashboardCategory getTilesForCategory(String key) {
|
|
return mCategoryManager.getTilesByCategory(mContext, key);
|
|
}
|
|
|
|
@Override
|
|
public List<DashboardCategory> getAllCategories() {
|
|
return mCategoryManager.getCategories(mContext);
|
|
}
|
|
|
|
@Override
|
|
public String getDashboardKeyForTile(Tile tile) {
|
|
if (tile == null) {
|
|
return null;
|
|
}
|
|
if (tile.hasKey()) {
|
|
return tile.getKey(mContext);
|
|
}
|
|
final StringBuilder sb = new StringBuilder(DASHBOARD_TILE_PREF_KEY_PREFIX);
|
|
final ComponentName component = tile.getIntent().getComponent();
|
|
sb.append(component.getClassName());
|
|
return sb.toString();
|
|
}
|
|
|
|
@Override
|
|
public List<DynamicDataObserver> bindPreferenceToTileAndGetObservers(FragmentActivity activity,
|
|
boolean forceRoundedIcon, int sourceMetricsCategory, Preference pref, Tile tile,
|
|
String key, int baseOrder) {
|
|
if (pref == null) {
|
|
return null;
|
|
}
|
|
if (!TextUtils.isEmpty(key)) {
|
|
pref.setKey(key);
|
|
} else {
|
|
pref.setKey(getDashboardKeyForTile(tile));
|
|
}
|
|
final List<DynamicDataObserver> outObservers = new ArrayList<>();
|
|
DynamicDataObserver observer = bindTitleAndGetObserver(pref, tile);
|
|
if (observer != null) {
|
|
outObservers.add(observer);
|
|
}
|
|
observer = bindSummaryAndGetObserver(pref, tile);
|
|
if (observer != null) {
|
|
outObservers.add(observer);
|
|
}
|
|
observer = bindSwitchAndGetObserver(pref, tile);
|
|
if (observer != null) {
|
|
outObservers.add(observer);
|
|
}
|
|
bindIcon(pref, tile, forceRoundedIcon);
|
|
|
|
if (tile instanceof ActivityTile) {
|
|
final Bundle metadata = tile.getMetaData();
|
|
String clsName = null;
|
|
String action = null;
|
|
if (metadata != null) {
|
|
clsName = metadata.getString(SettingsActivity.META_DATA_KEY_FRAGMENT_CLASS);
|
|
action = metadata.getString(META_DATA_KEY_INTENT_ACTION);
|
|
}
|
|
if (!TextUtils.isEmpty(clsName)) {
|
|
pref.setFragment(clsName);
|
|
} else {
|
|
final Intent intent = new Intent(tile.getIntent());
|
|
intent.putExtra(MetricsFeatureProvider.EXTRA_SOURCE_METRICS_CATEGORY,
|
|
sourceMetricsCategory);
|
|
if (action != null) {
|
|
intent.setAction(action);
|
|
}
|
|
pref.setOnPreferenceClickListener(preference -> {
|
|
launchIntentOrSelectProfile(activity, tile, intent, sourceMetricsCategory);
|
|
return true;
|
|
});
|
|
}
|
|
}
|
|
|
|
if (tile.hasOrder()) {
|
|
final String skipOffsetPackageName = activity.getPackageName();
|
|
final int order = tile.getOrder();
|
|
boolean shouldSkipBaseOrderOffset = TextUtils.equals(
|
|
skipOffsetPackageName, tile.getIntent().getComponent().getPackageName());
|
|
if (shouldSkipBaseOrderOffset || baseOrder == Preference.DEFAULT_ORDER) {
|
|
pref.setOrder(order);
|
|
} else {
|
|
pref.setOrder(order + baseOrder);
|
|
}
|
|
}
|
|
return outObservers.isEmpty() ? null : outObservers;
|
|
}
|
|
|
|
@Override
|
|
public void openTileIntent(FragmentActivity activity, Tile tile) {
|
|
if (tile == null) {
|
|
Intent intent = new Intent(Settings.ACTION_SETTINGS).addFlags(
|
|
Intent.FLAG_ACTIVITY_CLEAR_TASK);
|
|
mContext.startActivity(intent);
|
|
return;
|
|
}
|
|
final Intent intent = new Intent(tile.getIntent())
|
|
.putExtra(MetricsFeatureProvider.EXTRA_SOURCE_METRICS_CATEGORY,
|
|
SettingsEnums.DASHBOARD_SUMMARY)
|
|
.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TASK);
|
|
launchIntentOrSelectProfile(activity, tile, intent, SettingsEnums.DASHBOARD_SUMMARY);
|
|
}
|
|
|
|
private DynamicDataObserver createDynamicDataObserver(String method, Uri uri, Preference pref) {
|
|
return new DynamicDataObserver() {
|
|
@Override
|
|
public Uri getUri() {
|
|
return uri;
|
|
}
|
|
|
|
@Override
|
|
public void onDataChanged() {
|
|
switch (method) {
|
|
case METHOD_GET_DYNAMIC_TITLE:
|
|
refreshTitle(uri, pref);
|
|
break;
|
|
case METHOD_GET_DYNAMIC_SUMMARY:
|
|
refreshSummary(uri, pref);
|
|
break;
|
|
case METHOD_IS_CHECKED:
|
|
refreshSwitch(uri, pref);
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
private DynamicDataObserver bindTitleAndGetObserver(Preference preference, Tile tile) {
|
|
final CharSequence title = tile.getTitle(mContext.getApplicationContext());
|
|
if (title != null) {
|
|
preference.setTitle(title);
|
|
return null;
|
|
}
|
|
if (tile.getMetaData() != null && tile.getMetaData().containsKey(
|
|
META_DATA_PREFERENCE_TITLE_URI)) {
|
|
// Set a placeholder title before starting to fetch real title, this is necessary
|
|
// to avoid preference height change.
|
|
preference.setTitle(R.string.summary_placeholder);
|
|
|
|
final Uri uri = TileUtils.getCompleteUri(tile, META_DATA_PREFERENCE_TITLE_URI,
|
|
METHOD_GET_DYNAMIC_TITLE);
|
|
refreshTitle(uri, preference);
|
|
return createDynamicDataObserver(METHOD_GET_DYNAMIC_TITLE, uri, preference);
|
|
}
|
|
return null;
|
|
}
|
|
|
|
private void refreshTitle(Uri uri, Preference preference) {
|
|
ThreadUtils.postOnBackgroundThread(() -> {
|
|
final Map<String, IContentProvider> providerMap = new ArrayMap<>();
|
|
final String titleFromUri = TileUtils.getTextFromUri(
|
|
mContext, uri, providerMap, META_DATA_PREFERENCE_TITLE);
|
|
if (!TextUtils.equals(titleFromUri, preference.getTitle())) {
|
|
ThreadUtils.postOnMainThread(() -> preference.setTitle(titleFromUri));
|
|
}
|
|
});
|
|
}
|
|
|
|
private DynamicDataObserver bindSummaryAndGetObserver(Preference preference, Tile tile) {
|
|
final CharSequence summary = tile.getSummary(mContext);
|
|
if (summary != null) {
|
|
preference.setSummary(summary);
|
|
} else if (tile.getMetaData() != null
|
|
&& tile.getMetaData().containsKey(META_DATA_PREFERENCE_SUMMARY_URI)) {
|
|
// Set a placeholder summary before starting to fetch real summary, this is necessary
|
|
// to avoid preference height change.
|
|
preference.setSummary(R.string.summary_placeholder);
|
|
|
|
final Uri uri = TileUtils.getCompleteUri(tile, META_DATA_PREFERENCE_SUMMARY_URI,
|
|
METHOD_GET_DYNAMIC_SUMMARY);
|
|
refreshSummary(uri, preference);
|
|
return createDynamicDataObserver(METHOD_GET_DYNAMIC_SUMMARY, uri, preference);
|
|
}
|
|
return null;
|
|
}
|
|
|
|
private void refreshSummary(Uri uri, Preference preference) {
|
|
ThreadUtils.postOnBackgroundThread(() -> {
|
|
final Map<String, IContentProvider> providerMap = new ArrayMap<>();
|
|
final String summaryFromUri = TileUtils.getTextFromUri(
|
|
mContext, uri, providerMap, META_DATA_PREFERENCE_SUMMARY);
|
|
if (!TextUtils.equals(summaryFromUri, preference.getSummary())) {
|
|
ThreadUtils.postOnMainThread(() -> preference.setSummary(summaryFromUri));
|
|
}
|
|
});
|
|
}
|
|
|
|
private DynamicDataObserver bindSwitchAndGetObserver(Preference preference, Tile tile) {
|
|
if (!tile.hasSwitch()) {
|
|
return null;
|
|
}
|
|
|
|
final Uri onCheckedChangedUri = TileUtils.getCompleteUri(tile,
|
|
META_DATA_PREFERENCE_SWITCH_URI, METHOD_ON_CHECKED_CHANGED);
|
|
preference.setOnPreferenceChangeListener((pref, newValue) -> {
|
|
onCheckedChanged(onCheckedChangedUri, pref, (boolean) newValue);
|
|
return true;
|
|
});
|
|
|
|
final Uri isCheckedUri = TileUtils.getCompleteUri(tile, META_DATA_PREFERENCE_SWITCH_URI,
|
|
METHOD_IS_CHECKED);
|
|
setSwitchEnabled(preference, false);
|
|
refreshSwitch(isCheckedUri, preference);
|
|
return createDynamicDataObserver(METHOD_IS_CHECKED, isCheckedUri, preference);
|
|
}
|
|
|
|
private void onCheckedChanged(Uri uri, Preference pref, boolean checked) {
|
|
setSwitchEnabled(pref, false);
|
|
ThreadUtils.postOnBackgroundThread(() -> {
|
|
final Map<String, IContentProvider> providerMap = new ArrayMap<>();
|
|
final Bundle result = TileUtils.putBooleanToUriAndGetResult(mContext, uri, providerMap,
|
|
EXTRA_SWITCH_CHECKED_STATE, checked);
|
|
|
|
ThreadUtils.postOnMainThread(() -> {
|
|
setSwitchEnabled(pref, true);
|
|
final boolean error = result.getBoolean(EXTRA_SWITCH_SET_CHECKED_ERROR);
|
|
if (!error) {
|
|
return;
|
|
}
|
|
|
|
setSwitchChecked(pref, !checked);
|
|
final String errorMsg = result.getString(EXTRA_SWITCH_SET_CHECKED_ERROR_MESSAGE);
|
|
if (!TextUtils.isEmpty(errorMsg)) {
|
|
Toast.makeText(mContext, errorMsg, Toast.LENGTH_SHORT).show();
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
private void refreshSwitch(Uri uri, Preference preference) {
|
|
ThreadUtils.postOnBackgroundThread(() -> {
|
|
final Map<String, IContentProvider> providerMap = new ArrayMap<>();
|
|
final boolean checked = TileUtils.getBooleanFromUri(mContext, uri, providerMap,
|
|
EXTRA_SWITCH_CHECKED_STATE);
|
|
ThreadUtils.postOnMainThread(() -> {
|
|
setSwitchChecked(preference, checked);
|
|
setSwitchEnabled(preference, true);
|
|
});
|
|
});
|
|
}
|
|
|
|
private void setSwitchChecked(Preference pref, boolean checked) {
|
|
if (pref instanceof PrimarySwitchPreference) {
|
|
((PrimarySwitchPreference) pref).setChecked(checked);
|
|
} else if (pref instanceof SwitchPreference) {
|
|
((SwitchPreference) pref).setChecked(checked);
|
|
}
|
|
}
|
|
|
|
private void setSwitchEnabled(Preference pref, boolean enabled) {
|
|
if (pref instanceof PrimarySwitchPreference) {
|
|
((PrimarySwitchPreference) pref).setSwitchEnabled(enabled);
|
|
} else {
|
|
pref.setEnabled(enabled);
|
|
}
|
|
}
|
|
|
|
@VisibleForTesting
|
|
void bindIcon(Preference preference, Tile tile, boolean forceRoundedIcon) {
|
|
// Icon provided by the content provider overrides any static icon.
|
|
if (tile.getMetaData() != null
|
|
&& tile.getMetaData().containsKey(META_DATA_PREFERENCE_ICON_URI)) {
|
|
// Set a transparent color before starting to fetch the real icon, this is necessary
|
|
// to avoid preference padding change.
|
|
setPreferenceIcon(preference, tile, forceRoundedIcon, mContext.getPackageName(),
|
|
Icon.createWithResource(mContext, android.R.color.transparent));
|
|
|
|
ThreadUtils.postOnBackgroundThread(() -> {
|
|
final Intent intent = tile.getIntent();
|
|
String packageName = null;
|
|
if (!TextUtils.isEmpty(intent.getPackage())) {
|
|
packageName = intent.getPackage();
|
|
} else if (intent.getComponent() != null) {
|
|
packageName = intent.getComponent().getPackageName();
|
|
}
|
|
final Map<String, IContentProvider> providerMap = new ArrayMap<>();
|
|
final Uri uri = TileUtils.getCompleteUri(tile, META_DATA_PREFERENCE_ICON_URI,
|
|
METHOD_GET_PROVIDER_ICON);
|
|
final Pair<String, Integer> iconInfo = TileUtils.getIconFromUri(
|
|
mContext, packageName, uri, providerMap);
|
|
if (iconInfo == null) {
|
|
Log.w(TAG, "Failed to get icon from uri " + uri);
|
|
return;
|
|
}
|
|
final Icon icon = Icon.createWithResource(iconInfo.first, iconInfo.second);
|
|
ThreadUtils.postOnMainThread(() -> {
|
|
setPreferenceIcon(preference, tile, forceRoundedIcon, iconInfo.first, icon);
|
|
});
|
|
});
|
|
return;
|
|
}
|
|
|
|
// Use preference context instead here when get icon from Tile, as we are using the context
|
|
// to get the style to tint the icon. Using mContext here won't get the correct style.
|
|
final Icon tileIcon = tile.getIcon(preference.getContext());
|
|
if (tileIcon == null) {
|
|
return;
|
|
}
|
|
setPreferenceIcon(preference, tile, forceRoundedIcon, tile.getPackageName(), tileIcon);
|
|
}
|
|
|
|
private void setPreferenceIcon(Preference preference, Tile tile, boolean forceRoundedIcon,
|
|
String iconPackage, Icon icon) {
|
|
Drawable iconDrawable = icon.loadDrawable(preference.getContext());
|
|
if (TextUtils.equals(tile.getCategory(), CategoryKey.CATEGORY_HOMEPAGE)) {
|
|
iconDrawable.setTint(Utils.getHomepageIconColor(preference.getContext()));
|
|
} else if (forceRoundedIcon && !TextUtils.equals(mContext.getPackageName(), iconPackage)) {
|
|
iconDrawable = new AdaptiveIcon(mContext, iconDrawable,
|
|
R.dimen.dashboard_tile_foreground_image_inset);
|
|
((AdaptiveIcon) iconDrawable).setBackgroundColor(mContext, tile);
|
|
}
|
|
preference.setIcon(iconDrawable);
|
|
}
|
|
|
|
private void launchIntentOrSelectProfile(FragmentActivity activity, Tile tile, Intent intent,
|
|
int sourceMetricCategory) {
|
|
if (!isIntentResolvable(intent)) {
|
|
Log.w(TAG, "Cannot resolve intent, skipping. " + intent);
|
|
return;
|
|
}
|
|
ProfileSelectDialog.updateUserHandlesIfNeeded(mContext, tile);
|
|
mMetricsFeatureProvider.logStartedIntent(intent, sourceMetricCategory);
|
|
|
|
if (tile.userHandle == null || tile.isPrimaryProfileOnly()) {
|
|
activity.startActivity(intent);
|
|
} else if (tile.userHandle.size() == 1) {
|
|
activity.startActivityAsUser(intent, tile.userHandle.get(0));
|
|
} else {
|
|
final UserHandle userHandle = intent.getParcelableExtra(EXTRA_USER);
|
|
if (userHandle != null && tile.userHandle.contains(userHandle)) {
|
|
activity.startActivityAsUser(intent, userHandle);
|
|
return;
|
|
}
|
|
|
|
final List<UserHandle> resolvableUsers = getResolvableUsers(intent, tile);
|
|
if (resolvableUsers.size() == 1) {
|
|
activity.startActivityAsUser(intent, resolvableUsers.get(0));
|
|
return;
|
|
}
|
|
|
|
ProfileSelectDialog.show(activity.getSupportFragmentManager(), tile,
|
|
sourceMetricCategory);
|
|
}
|
|
}
|
|
|
|
private boolean isIntentResolvable(Intent intent) {
|
|
return mPackageManager.resolveActivity(intent, 0) != null;
|
|
}
|
|
|
|
private List<UserHandle> getResolvableUsers(Intent intent, Tile tile) {
|
|
final ArrayList<UserHandle> eligibleUsers = new ArrayList<>();
|
|
for (UserHandle user : tile.userHandle) {
|
|
if (mPackageManager.resolveActivityAsUser(intent, 0, user.getIdentifier()) != null) {
|
|
eligibleUsers.add(user);
|
|
}
|
|
}
|
|
return eligibleUsers;
|
|
}
|
|
}
|