Also add new SettingsSpaUnitTests. Bug: 236346018 Test: Manual with App Info page Test: atest SettingsSpaUnitTests Test: Manual compare generated Settings AndroidManifest.xml Change-Id: I9f6b2ca446fd3d196792a876a6e4049c5cf97a1d
820 lines
32 KiB
Java
Executable File
820 lines
32 KiB
Java
Executable File
/*
|
|
* Copyright (C) 2017 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.applications.appinfo;
|
|
|
|
import static android.app.admin.DevicePolicyResources.Strings.Settings.CONNECTED_WORK_AND_PERSONAL_APPS_TITLE;
|
|
|
|
import static com.android.settingslib.RestrictedLockUtils.EnforcedAdmin;
|
|
|
|
import android.app.Activity;
|
|
import android.app.AppOpsManager;
|
|
import android.app.KeyguardManager;
|
|
import android.app.admin.DevicePolicyManager;
|
|
import android.app.settings.SettingsEnums;
|
|
import android.content.BroadcastReceiver;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.IntentFilter;
|
|
import android.content.pm.ApplicationInfo;
|
|
import android.content.pm.PackageInfo;
|
|
import android.content.pm.PackageManager;
|
|
import android.content.pm.PackageManager.NameNotFoundException;
|
|
import android.content.pm.UserInfo;
|
|
import android.hardware.biometrics.BiometricManager;
|
|
import android.hardware.biometrics.BiometricPrompt;
|
|
import android.net.Uri;
|
|
import android.os.Bundle;
|
|
import android.os.CancellationSignal;
|
|
import android.os.Handler;
|
|
import android.os.Looper;
|
|
import android.os.UserHandle;
|
|
import android.os.UserManager;
|
|
import android.text.TextUtils;
|
|
import android.util.Log;
|
|
import android.view.Menu;
|
|
import android.view.MenuInflater;
|
|
import android.view.MenuItem;
|
|
import android.widget.Toast;
|
|
|
|
import androidx.annotation.VisibleForTesting;
|
|
|
|
import com.android.settings.R;
|
|
import com.android.settings.SettingsActivity;
|
|
import com.android.settings.SettingsPreferenceFragment;
|
|
import com.android.settings.applications.manageapplications.ManageApplications;
|
|
import com.android.settings.applications.mobilebundledapps.ApplicationMetadataUtils;
|
|
import com.android.settings.applications.specialaccess.interactacrossprofiles.InteractAcrossProfilesDetailsPreferenceController;
|
|
import com.android.settings.applications.specialaccess.pictureinpicture.PictureInPictureDetailPreferenceController;
|
|
import com.android.settings.core.SubSettingLauncher;
|
|
import com.android.settings.dashboard.DashboardFragment;
|
|
import com.android.settingslib.RestrictedLockUtilsInternal;
|
|
import com.android.settingslib.applications.AppUtils;
|
|
import com.android.settingslib.applications.ApplicationsState;
|
|
import com.android.settingslib.applications.ApplicationsState.AppEntry;
|
|
import com.android.settingslib.core.AbstractPreferenceController;
|
|
import com.android.settingslib.core.lifecycle.Lifecycle;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.Arrays;
|
|
import java.util.List;
|
|
|
|
/**
|
|
* Dashboard fragment to display application information from Settings. This activity presents
|
|
* extended information associated with a package like code, data, total size, permissions
|
|
* used by the application and also the set of default launchable activities.
|
|
* For system applications, an option to clear user data is displayed only if data size is > 0.
|
|
* System applications that do not want clear user data do not have this option.
|
|
* For non-system applications, there is no option to clear data. Instead there is an option to
|
|
* uninstall the application.
|
|
*/
|
|
public class AppInfoDashboardFragment extends DashboardFragment
|
|
implements ApplicationsState.Callbacks,
|
|
ButtonActionDialogFragment.AppButtonsDialogListener {
|
|
|
|
private static final String TAG = "AppInfoDashboard";
|
|
|
|
// Menu identifiers
|
|
@VisibleForTesting
|
|
static final int UNINSTALL_ALL_USERS_MENU = 1;
|
|
@VisibleForTesting
|
|
static final int UNINSTALL_UPDATES = 2;
|
|
static final int INSTALL_INSTANT_APP_MENU = 3;
|
|
static final int ACCESS_RESTRICTED_SETTINGS = 4;
|
|
|
|
// Result code identifiers
|
|
@VisibleForTesting
|
|
static final int REQUEST_UNINSTALL = 0;
|
|
private static final int REQUEST_REMOVE_DEVICE_ADMIN = 5;
|
|
|
|
static final int SUB_INFO_FRAGMENT = 1;
|
|
|
|
static final int LOADER_CHART_DATA = 2;
|
|
static final int LOADER_STORAGE = 3;
|
|
static final int LOADER_BATTERY = 4;
|
|
static final int LOADER_BATTERY_USAGE_STATS = 5;
|
|
|
|
public static final String ARG_PACKAGE_NAME = "package";
|
|
public static final String ARG_PACKAGE_UID = "uid";
|
|
|
|
private static final boolean localLOGV = false;
|
|
|
|
private EnforcedAdmin mAppsControlDisallowedAdmin;
|
|
private boolean mAppsControlDisallowedBySystem;
|
|
|
|
private ApplicationsState mState;
|
|
private ApplicationsState.Session mSession;
|
|
private ApplicationsState.AppEntry mAppEntry;
|
|
private PackageInfo mPackageInfo;
|
|
private int mUserId;
|
|
private String mPackageName;
|
|
private int mUid;
|
|
|
|
private DevicePolicyManager mDpm;
|
|
private UserManager mUserManager;
|
|
private PackageManager mPm;
|
|
|
|
@VisibleForTesting
|
|
boolean mFinishing;
|
|
private boolean mListeningToPackageRemove;
|
|
|
|
|
|
private boolean mInitialized;
|
|
private boolean mShowUninstalled;
|
|
private boolean mUpdatedSysApp = false;
|
|
|
|
private List<Callback> mCallbacks = new ArrayList<>();
|
|
|
|
private InstantAppButtonsPreferenceController mInstantAppButtonPreferenceController;
|
|
private AppButtonsPreferenceController mAppButtonsPreferenceController;
|
|
|
|
/**
|
|
* Callback to invoke when app info has been changed.
|
|
*/
|
|
public interface Callback {
|
|
void refreshUi();
|
|
}
|
|
|
|
@Override
|
|
public void onAttach(Context context) {
|
|
super.onAttach(context);
|
|
final String packageName = getPackageName();
|
|
final TimeSpentInAppPreferenceController timeSpentInAppPreferenceController = use(
|
|
TimeSpentInAppPreferenceController.class);
|
|
timeSpentInAppPreferenceController.setPackageName(packageName);
|
|
timeSpentInAppPreferenceController.setParentFragment(this);
|
|
timeSpentInAppPreferenceController.initLifeCycleOwner(this);
|
|
|
|
use(AppDataUsagePreferenceController.class).setParentFragment(this);
|
|
final AppInstallerInfoPreferenceController installer =
|
|
use(AppInstallerInfoPreferenceController.class);
|
|
installer.setPackageName(packageName);
|
|
installer.setParentFragment(this);
|
|
installer.setMbaWithMetadataStatus(ApplicationMetadataUtils.getDefaultInstance(),
|
|
packageName);
|
|
use(AppInstallerPreferenceCategoryController.class).setChildren(Arrays.asList(installer));
|
|
use(AppNotificationPreferenceController.class).setParentFragment(this);
|
|
|
|
use(AppOpenByDefaultPreferenceController.class)
|
|
.setPackageName(packageName)
|
|
.setParentFragment(this);
|
|
|
|
use(AppPermissionPreferenceController.class).setParentFragment(this);
|
|
use(AppPermissionPreferenceController.class).setPackageName(packageName);
|
|
use(AppSettingPreferenceController.class)
|
|
.setPackageName(packageName)
|
|
.setParentFragment(this);
|
|
use(AppAllServicesPreferenceController.class).setParentFragment(this);
|
|
use(AppAllServicesPreferenceController.class).setPackageName(packageName);
|
|
use(AppStoragePreferenceController.class).setParentFragment(this);
|
|
use(AppVersionPreferenceController.class).setParentFragment(this);
|
|
use(InstantAppDomainsPreferenceController.class).setParentFragment(this);
|
|
|
|
final HibernationSwitchPreferenceController appHibernationSettings =
|
|
use(HibernationSwitchPreferenceController.class);
|
|
appHibernationSettings.setParentFragment(this);
|
|
appHibernationSettings.setPackage(packageName);
|
|
use(AppHibernationPreferenceCategoryController.class).setChildren(
|
|
Arrays.asList(appHibernationSettings));
|
|
|
|
final WriteSystemSettingsPreferenceController writeSystemSettings =
|
|
use(WriteSystemSettingsPreferenceController.class);
|
|
writeSystemSettings.setParentFragment(this);
|
|
|
|
final DrawOverlayDetailPreferenceController drawOverlay =
|
|
use(DrawOverlayDetailPreferenceController.class);
|
|
drawOverlay.setParentFragment(this);
|
|
|
|
final PictureInPictureDetailPreferenceController pip =
|
|
use(PictureInPictureDetailPreferenceController.class);
|
|
pip.setPackageName(packageName);
|
|
pip.setParentFragment(this);
|
|
|
|
final ExternalSourceDetailPreferenceController externalSource =
|
|
use(ExternalSourceDetailPreferenceController.class);
|
|
externalSource.setPackageName(packageName);
|
|
externalSource.setParentFragment(this);
|
|
|
|
final InteractAcrossProfilesDetailsPreferenceController acrossProfiles =
|
|
use(InteractAcrossProfilesDetailsPreferenceController.class);
|
|
acrossProfiles.setPackageName(packageName);
|
|
acrossProfiles.setParentFragment(this);
|
|
|
|
final AlarmsAndRemindersDetailPreferenceController alarmsAndReminders =
|
|
use(AlarmsAndRemindersDetailPreferenceController.class);
|
|
alarmsAndReminders.setPackageName(packageName);
|
|
alarmsAndReminders.setParentFragment(this);
|
|
|
|
use(AdvancedAppInfoPreferenceCategoryController.class).setChildren(Arrays.asList(
|
|
writeSystemSettings, drawOverlay, pip, externalSource, acrossProfiles,
|
|
alarmsAndReminders));
|
|
|
|
final AppLocalePreferenceController appLocale =
|
|
use(AppLocalePreferenceController.class);
|
|
appLocale.setParentFragment(this);
|
|
}
|
|
|
|
@Override
|
|
public void onCreate(Bundle icicle) {
|
|
super.onCreate(icicle);
|
|
mFinishing = false;
|
|
final Activity activity = getActivity();
|
|
mDpm = (DevicePolicyManager) activity.getSystemService(Context.DEVICE_POLICY_SERVICE);
|
|
mUserManager = (UserManager) activity.getSystemService(Context.USER_SERVICE);
|
|
mPm = activity.getPackageManager();
|
|
if (!ensurePackageInfoAvailable(activity)) {
|
|
return;
|
|
}
|
|
if (!ensureDisplayableModule(activity)) {
|
|
return;
|
|
}
|
|
startListeningToPackageRemove();
|
|
|
|
setHasOptionsMenu(true);
|
|
replaceEnterpriseStringTitle("interact_across_profiles",
|
|
CONNECTED_WORK_AND_PERSONAL_APPS_TITLE, R.string.interact_across_profiles_title);
|
|
}
|
|
|
|
@Override
|
|
public void onCreatePreferences(Bundle savedInstanceState, String rootKey) {
|
|
if (!ensurePackageInfoAvailable(getActivity())) {
|
|
return;
|
|
}
|
|
super.onCreatePreferences(savedInstanceState, rootKey);
|
|
}
|
|
|
|
@Override
|
|
public void onDestroy() {
|
|
stopListeningToPackageRemove();
|
|
super.onDestroy();
|
|
}
|
|
|
|
@Override
|
|
public int getMetricsCategory() {
|
|
return SettingsEnums.APPLICATIONS_INSTALLED_APP_DETAILS;
|
|
}
|
|
|
|
@Override
|
|
public void onResume() {
|
|
super.onResume();
|
|
final Activity activity = getActivity();
|
|
mAppsControlDisallowedAdmin = RestrictedLockUtilsInternal.checkIfRestrictionEnforced(
|
|
activity, UserManager.DISALLOW_APPS_CONTROL, mUserId);
|
|
mAppsControlDisallowedBySystem = RestrictedLockUtilsInternal.hasBaseUserRestriction(
|
|
activity, UserManager.DISALLOW_APPS_CONTROL, mUserId);
|
|
|
|
if (!refreshUi()) {
|
|
setIntentAndFinish(true, true);
|
|
}
|
|
getActivity().invalidateOptionsMenu();
|
|
}
|
|
|
|
@Override
|
|
protected int getPreferenceScreenResId() {
|
|
return R.xml.app_info_settings;
|
|
}
|
|
|
|
@Override
|
|
protected String getLogTag() {
|
|
return TAG;
|
|
}
|
|
|
|
@Override
|
|
protected List<AbstractPreferenceController> createPreferenceControllers(Context context) {
|
|
retrieveAppEntry();
|
|
if (mPackageInfo == null) {
|
|
return null;
|
|
}
|
|
final String packageName = getPackageName();
|
|
final List<AbstractPreferenceController> controllers = new ArrayList<>();
|
|
final Lifecycle lifecycle = getSettingsLifecycle();
|
|
|
|
// The following are controllers for preferences that needs to refresh the preference state
|
|
// when app state changes.
|
|
controllers.add(
|
|
new AppHeaderViewPreferenceController(context, this, packageName, lifecycle));
|
|
|
|
for (AbstractPreferenceController controller : controllers) {
|
|
mCallbacks.add((Callback) controller);
|
|
}
|
|
|
|
// The following are controllers for preferences that don't need to refresh the preference
|
|
// state when app state changes.
|
|
mInstantAppButtonPreferenceController =
|
|
new InstantAppButtonsPreferenceController(context, this, packageName, lifecycle);
|
|
controllers.add(mInstantAppButtonPreferenceController);
|
|
mAppButtonsPreferenceController = new AppButtonsPreferenceController(
|
|
(SettingsActivity) getActivity(), this, lifecycle, packageName, mState,
|
|
REQUEST_UNINSTALL, REQUEST_REMOVE_DEVICE_ADMIN);
|
|
controllers.add(mAppButtonsPreferenceController);
|
|
controllers.add(new AppBatteryPreferenceController(
|
|
context, this, packageName, getUid(), lifecycle));
|
|
controllers.add(new AppMemoryPreferenceController(context, this, lifecycle));
|
|
controllers.add(new DefaultHomeShortcutPreferenceController(context, packageName));
|
|
controllers.add(new DefaultBrowserShortcutPreferenceController(context, packageName));
|
|
controllers.add(new DefaultPhoneShortcutPreferenceController(context, packageName));
|
|
controllers.add(new DefaultEmergencyShortcutPreferenceController(context, packageName));
|
|
controllers.add(new DefaultSmsShortcutPreferenceController(context, packageName));
|
|
|
|
return controllers;
|
|
}
|
|
|
|
void addToCallbackList(Callback callback) {
|
|
if (callback != null) {
|
|
mCallbacks.add(callback);
|
|
}
|
|
}
|
|
|
|
ApplicationsState.AppEntry getAppEntry() {
|
|
return mAppEntry;
|
|
}
|
|
|
|
void setAppEntry(ApplicationsState.AppEntry appEntry) {
|
|
mAppEntry = appEntry;
|
|
}
|
|
|
|
public PackageInfo getPackageInfo() {
|
|
return mPackageInfo;
|
|
}
|
|
|
|
@Override
|
|
public void onPackageSizeChanged(String packageName) {
|
|
if (!TextUtils.equals(packageName, mPackageName)) {
|
|
Log.d(TAG, "Package change irrelevant, skipping");
|
|
return;
|
|
}
|
|
refreshUi();
|
|
}
|
|
|
|
/**
|
|
* Ensures the {@link PackageInfo} is available to proceed. If it's not available, the fragment
|
|
* will finish.
|
|
*
|
|
* @return true if packageInfo is available.
|
|
*/
|
|
@VisibleForTesting
|
|
boolean ensurePackageInfoAvailable(Activity activity) {
|
|
if (mPackageInfo == null) {
|
|
mFinishing = true;
|
|
Log.w(TAG, "Package info not available. Is this package already uninstalled?");
|
|
activity.finishAndRemoveTask();
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Ensures the package is displayable as directed by {@link AppUtils#isHiddenSystemModule}.
|
|
* If it's not, the fragment will finish.
|
|
*
|
|
* @return true if package is displayable.
|
|
*/
|
|
@VisibleForTesting
|
|
boolean ensureDisplayableModule(Activity activity) {
|
|
if (AppUtils.isHiddenSystemModule(activity.getApplicationContext(), mPackageName)) {
|
|
mFinishing = true;
|
|
Log.w(TAG, "Package is hidden module, exiting: " + mPackageName);
|
|
activity.finishAndRemoveTask();
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
|
|
super.onCreateOptionsMenu(menu, inflater);
|
|
menu.add(0, UNINSTALL_UPDATES, 0, R.string.app_factory_reset)
|
|
.setShowAsAction(MenuItem.SHOW_AS_ACTION_NEVER);
|
|
menu.add(0, UNINSTALL_ALL_USERS_MENU, 1, R.string.uninstall_all_users_text)
|
|
.setShowAsAction(MenuItem.SHOW_AS_ACTION_NEVER);
|
|
menu.add(0, ACCESS_RESTRICTED_SETTINGS, 0,
|
|
R.string.app_restricted_settings_lockscreen_title)
|
|
.setShowAsAction(MenuItem.SHOW_AS_ACTION_NEVER);
|
|
}
|
|
|
|
@Override
|
|
public void onPrepareOptionsMenu(Menu menu) {
|
|
if (mFinishing) {
|
|
return;
|
|
}
|
|
super.onPrepareOptionsMenu(menu);
|
|
menu.findItem(UNINSTALL_ALL_USERS_MENU).setVisible(shouldShowUninstallForAll(mAppEntry));
|
|
menu.findItem(ACCESS_RESTRICTED_SETTINGS).setVisible(shouldShowAccessRestrictedSettings());
|
|
mUpdatedSysApp = (mAppEntry.info.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0;
|
|
final MenuItem uninstallUpdatesItem = menu.findItem(UNINSTALL_UPDATES);
|
|
final boolean uninstallUpdateDisabled = getContext().getResources().getBoolean(
|
|
R.bool.config_disable_uninstall_update);
|
|
uninstallUpdatesItem.setVisible(mUserManager.isAdminUser()
|
|
&& mUpdatedSysApp
|
|
&& !mAppsControlDisallowedBySystem
|
|
&& !uninstallUpdateDisabled);
|
|
if (uninstallUpdatesItem.isVisible()) {
|
|
RestrictedLockUtilsInternal.setMenuItemAsDisabledByAdmin(getActivity(),
|
|
uninstallUpdatesItem, mAppsControlDisallowedAdmin);
|
|
}
|
|
}
|
|
|
|
private static void showLockScreen(Context context, Runnable successRunnable) {
|
|
final KeyguardManager keyguardManager = context.getSystemService(
|
|
KeyguardManager.class);
|
|
|
|
if (keyguardManager.isKeyguardSecure()) {
|
|
final BiometricPrompt.AuthenticationCallback authenticationCallback =
|
|
new BiometricPrompt.AuthenticationCallback() {
|
|
@Override
|
|
public void onAuthenticationSucceeded(
|
|
BiometricPrompt.AuthenticationResult result) {
|
|
successRunnable.run();
|
|
}
|
|
|
|
@Override
|
|
public void onAuthenticationError(int errorCode, CharSequence errString) {
|
|
//Do nothing
|
|
}
|
|
};
|
|
|
|
final BiometricPrompt.Builder builder = new BiometricPrompt.Builder(context)
|
|
.setUseDefaultTitle(); // use default title if title is null/empty
|
|
|
|
final BiometricManager bm = context.getSystemService(BiometricManager.class);
|
|
final int authenticators = BiometricManager.Authenticators.DEVICE_CREDENTIAL
|
|
| BiometricManager.Authenticators.BIOMETRIC_WEAK;
|
|
if (bm.canAuthenticate(authenticators) == BiometricManager.BIOMETRIC_SUCCESS) {
|
|
builder.setAllowedAuthenticators(authenticators);
|
|
builder.setSubtitle(bm.getStrings(authenticators).getPromptMessage());
|
|
}
|
|
|
|
final BiometricPrompt bp = builder.build();
|
|
final Handler handler = new Handler(Looper.getMainLooper());
|
|
bp.authenticate(new CancellationSignal(),
|
|
runnable -> handler.post(runnable),
|
|
authenticationCallback);
|
|
} else {
|
|
successRunnable.run();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean onOptionsItemSelected(MenuItem item) {
|
|
switch (item.getItemId()) {
|
|
case UNINSTALL_ALL_USERS_MENU:
|
|
uninstallPkg(mAppEntry.info.packageName, true, false);
|
|
return true;
|
|
case UNINSTALL_UPDATES:
|
|
uninstallPkg(mAppEntry.info.packageName, false, false);
|
|
return true;
|
|
case ACCESS_RESTRICTED_SETTINGS:
|
|
showLockScreen(getContext(), () -> {
|
|
final AppOpsManager appOpsManager = getContext().getSystemService(
|
|
AppOpsManager.class);
|
|
appOpsManager.setMode(AppOpsManager.OP_ACCESS_RESTRICTED_SETTINGS,
|
|
getUid(),
|
|
getPackageName(),
|
|
AppOpsManager.MODE_ALLOWED);
|
|
getActivity().invalidateOptionsMenu();
|
|
final String toastString = getContext().getString(
|
|
R.string.toast_allows_restricted_settings_successfully,
|
|
mAppEntry.label);
|
|
Toast.makeText(getContext(), toastString, Toast.LENGTH_LONG).show();
|
|
});
|
|
return true;
|
|
}
|
|
return super.onOptionsItemSelected(item);
|
|
}
|
|
|
|
@Override
|
|
public void onActivityResult(int requestCode, int resultCode, Intent data) {
|
|
super.onActivityResult(requestCode, resultCode, data);
|
|
if (requestCode == REQUEST_UNINSTALL) {
|
|
// Refresh option menu
|
|
getActivity().invalidateOptionsMenu();
|
|
}
|
|
if (mAppButtonsPreferenceController != null) {
|
|
mAppButtonsPreferenceController.handleActivityResult(requestCode, resultCode, data);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void handleDialogClick(int id) {
|
|
if (mAppButtonsPreferenceController != null) {
|
|
mAppButtonsPreferenceController.handleDialogClick(id);
|
|
}
|
|
}
|
|
|
|
private boolean shouldShowAccessRestrictedSettings() {
|
|
try {
|
|
final int mode = getSystemService(AppOpsManager.class).noteOpNoThrow(
|
|
AppOpsManager.OP_ACCESS_RESTRICTED_SETTINGS, getUid(),
|
|
getPackageName());
|
|
return mode == AppOpsManager.MODE_IGNORED;
|
|
} catch (Exception e) {
|
|
// Fallback in case if app ops is not available in testing.
|
|
return false;
|
|
}
|
|
}
|
|
|
|
@VisibleForTesting
|
|
boolean shouldShowUninstallForAll(AppEntry appEntry) {
|
|
boolean showIt = true;
|
|
if (mUpdatedSysApp) {
|
|
showIt = false;
|
|
} else if (appEntry == null) {
|
|
showIt = false;
|
|
} else if ((appEntry.info.flags & ApplicationInfo.FLAG_SYSTEM) != 0) {
|
|
showIt = false;
|
|
} else if (mPackageInfo == null || mDpm.packageHasActiveAdmins(mPackageInfo.packageName)) {
|
|
showIt = false;
|
|
} else if (UserHandle.myUserId() != 0) {
|
|
showIt = false;
|
|
} else if (mUserManager.getUsers().size() < 2) {
|
|
showIt = false;
|
|
} else if (getNumberOfUserWithPackageInstalled(mPackageName) < 2
|
|
&& (appEntry.info.flags & ApplicationInfo.FLAG_INSTALLED) != 0) {
|
|
showIt = false;
|
|
} else if (AppUtils.isInstant(appEntry.info)) {
|
|
showIt = false;
|
|
}
|
|
return showIt;
|
|
}
|
|
|
|
@VisibleForTesting
|
|
boolean refreshUi() {
|
|
retrieveAppEntry();
|
|
if (mAppEntry == null) {
|
|
return false; // onCreate must have failed, make sure to exit
|
|
}
|
|
|
|
if (mPackageInfo == null) {
|
|
return false; // onCreate must have failed, make sure to exit
|
|
}
|
|
|
|
mState.ensureIcon(mAppEntry);
|
|
|
|
// Update the preference summaries.
|
|
for (Callback callback : mCallbacks) {
|
|
callback.refreshUi();
|
|
}
|
|
if (mAppButtonsPreferenceController.isAvailable()) {
|
|
mAppButtonsPreferenceController.refreshUi();
|
|
}
|
|
|
|
if (!mInitialized) {
|
|
// First time init: are we displaying an uninstalled app?
|
|
mInitialized = true;
|
|
mShowUninstalled = (mAppEntry.info.flags & ApplicationInfo.FLAG_INSTALLED) == 0;
|
|
} else {
|
|
// All other times: if the app no longer exists then we want
|
|
// to go away.
|
|
try {
|
|
final ApplicationInfo ainfo = getActivity().getPackageManager().getApplicationInfo(
|
|
mAppEntry.info.packageName,
|
|
PackageManager.MATCH_DISABLED_COMPONENTS
|
|
| PackageManager.MATCH_ANY_USER);
|
|
if (!mShowUninstalled) {
|
|
// If we did not start out with the app uninstalled, then
|
|
// it transitioning to the uninstalled state for the current
|
|
// user means we should go away as well.
|
|
return (ainfo.flags & ApplicationInfo.FLAG_INSTALLED) != 0;
|
|
}
|
|
} catch (NameNotFoundException e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
protected boolean shouldSkipForInitialSUW() {
|
|
return true;
|
|
}
|
|
|
|
private void uninstallPkg(String packageName, boolean allUsers, boolean andDisable) {
|
|
stopListeningToPackageRemove();
|
|
// Create new intent to launch Uninstaller activity
|
|
final Uri packageURI = Uri.parse("package:" + packageName);
|
|
final Intent uninstallIntent = new Intent(Intent.ACTION_UNINSTALL_PACKAGE, packageURI);
|
|
uninstallIntent.putExtra(Intent.EXTRA_UNINSTALL_ALL_USERS, allUsers);
|
|
mMetricsFeatureProvider.action(
|
|
getContext(), SettingsEnums.ACTION_SETTINGS_UNINSTALL_APP);
|
|
startActivityForResult(uninstallIntent, REQUEST_UNINSTALL);
|
|
}
|
|
|
|
public static void startAppInfoFragment(Class<?> fragment, int title, Bundle args,
|
|
SettingsPreferenceFragment caller, AppEntry appEntry) {
|
|
// start new fragment to display extended information
|
|
if (args == null) {
|
|
args = new Bundle();
|
|
}
|
|
args.putString(ARG_PACKAGE_NAME, appEntry.info.packageName);
|
|
args.putInt(ARG_PACKAGE_UID, appEntry.info.uid);
|
|
new SubSettingLauncher(caller.getContext())
|
|
.setDestination(fragment.getName())
|
|
.setArguments(args)
|
|
.setTitleRes(title)
|
|
.setResultListener(caller, SUB_INFO_FRAGMENT)
|
|
.setSourceMetricsCategory(caller.getMetricsCategory())
|
|
.launch();
|
|
}
|
|
|
|
/** Starts app info fragment from SPA pages. */
|
|
public static void startAppInfoFragment(
|
|
Class<?> destination, ApplicationInfo app, Context context, int sourceMetricsCategory) {
|
|
// start new fragment to display extended information
|
|
Bundle args = new Bundle();
|
|
args.putString(ARG_PACKAGE_NAME, app.packageName);
|
|
args.putInt(ARG_PACKAGE_UID, app.uid);
|
|
new SubSettingLauncher(context)
|
|
.setDestination(destination.getName())
|
|
.setArguments(args)
|
|
.setSourceMetricsCategory(sourceMetricsCategory)
|
|
.launch();
|
|
}
|
|
|
|
private void onPackageRemoved() {
|
|
getActivity().finishActivity(SUB_INFO_FRAGMENT);
|
|
getActivity().finishAndRemoveTask();
|
|
}
|
|
|
|
@VisibleForTesting
|
|
int getNumberOfUserWithPackageInstalled(String packageName) {
|
|
final List<UserInfo> userInfos = mUserManager.getAliveUsers();
|
|
int count = 0;
|
|
|
|
for (final UserInfo userInfo : userInfos) {
|
|
try {
|
|
// Use this API to check whether user has this package
|
|
final ApplicationInfo info = mPm.getApplicationInfoAsUser(
|
|
packageName, PackageManager.GET_META_DATA, userInfo.id);
|
|
if ((info.flags & ApplicationInfo.FLAG_INSTALLED) != 0) {
|
|
count++;
|
|
}
|
|
} catch (NameNotFoundException e) {
|
|
Log.e(TAG, "Package: " + packageName + " not found for user: " + userInfo.id);
|
|
}
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
private String getPackageName() {
|
|
if (mPackageName != null) {
|
|
return mPackageName;
|
|
}
|
|
final Bundle args = getArguments();
|
|
mPackageName = (args != null) ? args.getString(ARG_PACKAGE_NAME) : null;
|
|
if (mPackageName == null) {
|
|
final Intent intent = args == null ?
|
|
getActivity().getIntent() : (Intent) args.getParcelable("intent");
|
|
if (intent != null) {
|
|
mPackageName = intent.getData().getSchemeSpecificPart();
|
|
}
|
|
}
|
|
return mPackageName;
|
|
}
|
|
|
|
private int getUid() {
|
|
if (mUid > 0) {
|
|
return mUid;
|
|
}
|
|
final Bundle args = getArguments();
|
|
mUid = (args != null) ? args.getInt(ARG_PACKAGE_UID) : -1;
|
|
if (mUid <= 0) {
|
|
final Intent intent = args == null
|
|
? getActivity().getIntent() : (Intent) args.getParcelable("intent");
|
|
mUid = intent != null && intent.getExtras() != null
|
|
? mUid = intent.getIntExtra("uId", -1) : -1;
|
|
}
|
|
return mUid;
|
|
}
|
|
|
|
@VisibleForTesting
|
|
void retrieveAppEntry() {
|
|
final Activity activity = getActivity();
|
|
if (activity == null || mFinishing) {
|
|
return;
|
|
}
|
|
if (mState == null) {
|
|
mState = ApplicationsState.getInstance(activity.getApplication());
|
|
mSession = mState.newSession(this, getSettingsLifecycle());
|
|
}
|
|
mUserId = UserHandle.myUserId();
|
|
mAppEntry = mState.getEntry(getPackageName(), UserHandle.myUserId());
|
|
if (mAppEntry != null) {
|
|
// Get application info again to refresh changed properties of application
|
|
try {
|
|
mPackageInfo = activity.getPackageManager().getPackageInfo(
|
|
mAppEntry.info.packageName,
|
|
PackageManager.MATCH_DISABLED_COMPONENTS |
|
|
PackageManager.MATCH_ANY_USER |
|
|
PackageManager.GET_SIGNATURES |
|
|
PackageManager.GET_PERMISSIONS);
|
|
} catch (NameNotFoundException e) {
|
|
Log.e(TAG, "Exception when retrieving package:" + mAppEntry.info.packageName, e);
|
|
}
|
|
} else {
|
|
Log.w(TAG, "Missing AppEntry; maybe reinstalling?");
|
|
mPackageInfo = null;
|
|
}
|
|
}
|
|
|
|
private void setIntentAndFinish(boolean finish, boolean appChanged) {
|
|
if (localLOGV) Log.i(TAG, "appChanged=" + appChanged);
|
|
final Intent intent = new Intent();
|
|
intent.putExtra(ManageApplications.APP_CHG, appChanged);
|
|
final SettingsActivity sa = (SettingsActivity) getActivity();
|
|
sa.finishPreferencePanel(Activity.RESULT_OK, intent);
|
|
mFinishing = true;
|
|
}
|
|
|
|
@Override
|
|
public void onRunningStateChanged(boolean running) {
|
|
// No op.
|
|
}
|
|
|
|
@Override
|
|
public void onRebuildComplete(ArrayList<AppEntry> apps) {
|
|
// No op.
|
|
}
|
|
|
|
@Override
|
|
public void onPackageIconChanged() {
|
|
// No op.
|
|
}
|
|
|
|
@Override
|
|
public void onAllSizesComputed() {
|
|
// No op.
|
|
}
|
|
|
|
@Override
|
|
public void onLauncherInfoChanged() {
|
|
// No op.
|
|
}
|
|
|
|
@Override
|
|
public void onLoadEntriesCompleted() {
|
|
// No op.
|
|
}
|
|
|
|
@Override
|
|
public void onPackageListChanged() {
|
|
if (!refreshUi()) {
|
|
setIntentAndFinish(true, true);
|
|
}
|
|
}
|
|
|
|
@VisibleForTesting
|
|
void startListeningToPackageRemove() {
|
|
if (mListeningToPackageRemove) {
|
|
return;
|
|
}
|
|
mListeningToPackageRemove = true;
|
|
final IntentFilter filter = new IntentFilter(Intent.ACTION_PACKAGE_REMOVED);
|
|
filter.addDataScheme("package");
|
|
getContext().registerReceiver(mPackageRemovedReceiver, filter);
|
|
}
|
|
|
|
private void stopListeningToPackageRemove() {
|
|
if (!mListeningToPackageRemove) {
|
|
return;
|
|
}
|
|
mListeningToPackageRemove = false;
|
|
getContext().unregisterReceiver(mPackageRemovedReceiver);
|
|
}
|
|
|
|
@VisibleForTesting
|
|
final BroadcastReceiver mPackageRemovedReceiver = new BroadcastReceiver() {
|
|
@Override
|
|
public void onReceive(Context context, Intent intent) {
|
|
if (mFinishing) {
|
|
return;
|
|
}
|
|
|
|
final String packageName = intent.getData().getSchemeSpecificPart();
|
|
if (mAppEntry == null
|
|
|| mAppEntry.info == null
|
|
|| TextUtils.equals(mAppEntry.info.packageName, packageName)) {
|
|
onPackageRemoved();
|
|
} else if (mAppEntry.info.isResourceOverlay()
|
|
&& TextUtils.equals(mPackageInfo.overlayTarget, packageName)) {
|
|
refreshUi();
|
|
}
|
|
}
|
|
};
|
|
|
|
}
|