* commit 'a83515813cb5664fd68f583caa6d0bc6b6bc0042': Prohibit uninstall of the default (or only) home app
1429 lines
62 KiB
Java
Executable File
1429 lines
62 KiB
Java
Executable File
/**
|
|
* Copyright (C) 2007 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;
|
|
|
|
import com.android.internal.telephony.ISms;
|
|
import com.android.internal.telephony.SmsUsageMonitor;
|
|
import com.android.settings.R;
|
|
import com.android.settings.SettingsActivity;
|
|
import com.android.settings.Utils;
|
|
import com.android.settings.applications.ApplicationsState.AppEntry;
|
|
|
|
import android.app.Activity;
|
|
import android.app.ActivityManager;
|
|
import android.app.AlertDialog;
|
|
import android.app.Dialog;
|
|
import android.app.DialogFragment;
|
|
import android.app.Fragment;
|
|
import android.app.INotificationManager;
|
|
import android.app.admin.DevicePolicyManager;
|
|
import android.appwidget.AppWidgetManager;
|
|
import android.content.BroadcastReceiver;
|
|
import android.content.ComponentName;
|
|
import android.content.Context;
|
|
import android.content.DialogInterface;
|
|
import android.content.Intent;
|
|
import android.content.IntentFilter;
|
|
import android.content.pm.ApplicationInfo;
|
|
import android.content.pm.IPackageDataObserver;
|
|
import android.content.pm.IPackageMoveObserver;
|
|
import android.content.pm.PackageInfo;
|
|
import android.content.pm.PackageManager;
|
|
import android.content.pm.ResolveInfo;
|
|
import android.content.pm.PackageManager.NameNotFoundException;
|
|
import android.content.res.Resources;
|
|
import android.hardware.usb.IUsbManager;
|
|
import android.net.Uri;
|
|
import android.os.AsyncTask;
|
|
import android.os.Bundle;
|
|
import android.os.Environment;
|
|
import android.os.Handler;
|
|
import android.os.IBinder;
|
|
import android.os.Message;
|
|
import android.os.RemoteException;
|
|
import android.os.ServiceManager;
|
|
import android.os.UserHandle;
|
|
import android.os.UserManager;
|
|
import android.text.SpannableString;
|
|
import android.text.TextUtils;
|
|
import android.text.format.Formatter;
|
|
import android.text.style.BulletSpan;
|
|
import android.util.Log;
|
|
|
|
import java.lang.ref.WeakReference;
|
|
import java.util.ArrayList;
|
|
import java.util.HashSet;
|
|
import java.util.List;
|
|
import android.view.LayoutInflater;
|
|
import android.view.Menu;
|
|
import android.view.MenuInflater;
|
|
import android.view.MenuItem;
|
|
import android.view.View;
|
|
import android.view.ViewGroup;
|
|
import android.widget.AdapterView;
|
|
import android.widget.AppSecurityPermissions;
|
|
import android.widget.ArrayAdapter;
|
|
import android.widget.Button;
|
|
import android.widget.CheckBox;
|
|
import android.widget.CompoundButton;
|
|
import android.widget.ImageView;
|
|
import android.widget.LinearLayout;
|
|
import android.widget.Spinner;
|
|
import android.widget.TextView;
|
|
|
|
/**
|
|
* Activity 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 InstalledAppDetails extends Fragment
|
|
implements View.OnClickListener, CompoundButton.OnCheckedChangeListener,
|
|
ApplicationsState.Callbacks {
|
|
private static final String TAG="InstalledAppDetails";
|
|
private static final boolean localLOGV = false;
|
|
|
|
public static final String ARG_PACKAGE_NAME = "package";
|
|
|
|
private PackageManager mPm;
|
|
private UserManager mUserManager;
|
|
private IUsbManager mUsbManager;
|
|
private AppWidgetManager mAppWidgetManager;
|
|
private DevicePolicyManager mDpm;
|
|
private ISms mSmsManager;
|
|
private ApplicationsState mState;
|
|
private ApplicationsState.Session mSession;
|
|
private ApplicationsState.AppEntry mAppEntry;
|
|
private boolean mInitialized;
|
|
private boolean mShowUninstalled;
|
|
private PackageInfo mPackageInfo;
|
|
private CanBeOnSdCardChecker mCanBeOnSdCardChecker;
|
|
private View mRootView;
|
|
private Button mUninstallButton;
|
|
private View mMoreControlButtons;
|
|
private Button mSpecialDisableButton;
|
|
private boolean mMoveInProgress = false;
|
|
private boolean mUpdatedSysApp = false;
|
|
private Button mActivitiesButton;
|
|
private View mScreenCompatSection;
|
|
private CheckBox mAskCompatibilityCB;
|
|
private CheckBox mEnableCompatibilityCB;
|
|
private boolean mCanClearData = true;
|
|
private TextView mAppVersion;
|
|
private TextView mTotalSize;
|
|
private TextView mAppSize;
|
|
private TextView mDataSize;
|
|
private TextView mExternalCodeSize;
|
|
private TextView mExternalDataSize;
|
|
private ClearUserDataObserver mClearDataObserver;
|
|
// Views related to cache info
|
|
private TextView mCacheSize;
|
|
private Button mClearCacheButton;
|
|
private ClearCacheObserver mClearCacheObserver;
|
|
private Button mForceStopButton;
|
|
private Button mClearDataButton;
|
|
private Button mMoveAppButton;
|
|
private CompoundButton mNotificationSwitch;
|
|
|
|
private PackageMoveObserver mPackageMoveObserver;
|
|
|
|
private final HashSet<String> mHomePackages = new HashSet<String>();
|
|
|
|
private boolean mDisableAfterUninstall;
|
|
|
|
private boolean mHaveSizes = false;
|
|
private long mLastCodeSize = -1;
|
|
private long mLastDataSize = -1;
|
|
private long mLastExternalCodeSize = -1;
|
|
private long mLastExternalDataSize = -1;
|
|
private long mLastCacheSize = -1;
|
|
private long mLastTotalSize = -1;
|
|
|
|
//internal constants used in Handler
|
|
private static final int OP_SUCCESSFUL = 1;
|
|
private static final int OP_FAILED = 2;
|
|
private static final int CLEAR_USER_DATA = 1;
|
|
private static final int CLEAR_CACHE = 3;
|
|
private static final int PACKAGE_MOVE = 4;
|
|
|
|
// invalid size value used initially and also when size retrieval through PackageManager
|
|
// fails for whatever reason
|
|
private static final int SIZE_INVALID = -1;
|
|
|
|
// Resource strings
|
|
private CharSequence mInvalidSizeStr;
|
|
private CharSequence mComputingStr;
|
|
|
|
// Dialog identifiers used in showDialog
|
|
private static final int DLG_BASE = 0;
|
|
private static final int DLG_CLEAR_DATA = DLG_BASE + 1;
|
|
private static final int DLG_FACTORY_RESET = DLG_BASE + 2;
|
|
private static final int DLG_APP_NOT_FOUND = DLG_BASE + 3;
|
|
private static final int DLG_CANNOT_CLEAR_DATA = DLG_BASE + 4;
|
|
private static final int DLG_FORCE_STOP = DLG_BASE + 5;
|
|
private static final int DLG_MOVE_FAILED = DLG_BASE + 6;
|
|
private static final int DLG_DISABLE = DLG_BASE + 7;
|
|
private static final int DLG_DISABLE_NOTIFICATIONS = DLG_BASE + 8;
|
|
private static final int DLG_SPECIAL_DISABLE = DLG_BASE + 9;
|
|
|
|
// Menu identifiers
|
|
public static final int UNINSTALL_ALL_USERS_MENU = 1;
|
|
|
|
// Result code identifiers
|
|
public static final int REQUEST_UNINSTALL = 1;
|
|
public static final int REQUEST_MANAGE_SPACE = 2;
|
|
|
|
private Handler mHandler = new Handler() {
|
|
public void handleMessage(Message msg) {
|
|
// If the fragment is gone, don't process any more messages.
|
|
if (getView() == null) {
|
|
return;
|
|
}
|
|
switch (msg.what) {
|
|
case CLEAR_USER_DATA:
|
|
processClearMsg(msg);
|
|
break;
|
|
case CLEAR_CACHE:
|
|
// Refresh size info
|
|
mState.requestSize(mAppEntry.info.packageName);
|
|
break;
|
|
case PACKAGE_MOVE:
|
|
processMoveMsg(msg);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
|
|
class ClearUserDataObserver extends IPackageDataObserver.Stub {
|
|
public void onRemoveCompleted(final String packageName, final boolean succeeded) {
|
|
final Message msg = mHandler.obtainMessage(CLEAR_USER_DATA);
|
|
msg.arg1 = succeeded?OP_SUCCESSFUL:OP_FAILED;
|
|
mHandler.sendMessage(msg);
|
|
}
|
|
}
|
|
|
|
class ClearCacheObserver extends IPackageDataObserver.Stub {
|
|
public void onRemoveCompleted(final String packageName, final boolean succeeded) {
|
|
final Message msg = mHandler.obtainMessage(CLEAR_CACHE);
|
|
msg.arg1 = succeeded ? OP_SUCCESSFUL:OP_FAILED;
|
|
mHandler.sendMessage(msg);
|
|
}
|
|
}
|
|
|
|
class PackageMoveObserver extends IPackageMoveObserver.Stub {
|
|
public void packageMoved(String packageName, int returnCode) throws RemoteException {
|
|
final Message msg = mHandler.obtainMessage(PACKAGE_MOVE);
|
|
msg.arg1 = returnCode;
|
|
mHandler.sendMessage(msg);
|
|
}
|
|
}
|
|
|
|
private String getSizeStr(long size) {
|
|
if (size == SIZE_INVALID) {
|
|
return mInvalidSizeStr.toString();
|
|
}
|
|
return Formatter.formatFileSize(getActivity(), size);
|
|
}
|
|
|
|
private void initDataButtons() {
|
|
// If the app doesn't have its own space management UI
|
|
// And it's a system app that doesn't allow clearing user data or is an active admin
|
|
// Then disable the Clear Data button.
|
|
if (mAppEntry.info.manageSpaceActivityName == null
|
|
&& ((mAppEntry.info.flags&(ApplicationInfo.FLAG_SYSTEM
|
|
| ApplicationInfo.FLAG_ALLOW_CLEAR_USER_DATA))
|
|
== ApplicationInfo.FLAG_SYSTEM
|
|
|| mDpm.packageHasActiveAdmins(mPackageInfo.packageName))) {
|
|
mClearDataButton.setText(R.string.clear_user_data_text);
|
|
mClearDataButton.setEnabled(false);
|
|
mCanClearData = false;
|
|
} else {
|
|
if (mAppEntry.info.manageSpaceActivityName != null) {
|
|
mClearDataButton.setText(R.string.manage_space_text);
|
|
} else {
|
|
mClearDataButton.setText(R.string.clear_user_data_text);
|
|
}
|
|
mClearDataButton.setOnClickListener(this);
|
|
}
|
|
}
|
|
|
|
private CharSequence getMoveErrMsg(int errCode) {
|
|
switch (errCode) {
|
|
case PackageManager.MOVE_FAILED_INSUFFICIENT_STORAGE:
|
|
return getActivity().getString(R.string.insufficient_storage);
|
|
case PackageManager.MOVE_FAILED_DOESNT_EXIST:
|
|
return getActivity().getString(R.string.does_not_exist);
|
|
case PackageManager.MOVE_FAILED_FORWARD_LOCKED:
|
|
return getActivity().getString(R.string.app_forward_locked);
|
|
case PackageManager.MOVE_FAILED_INVALID_LOCATION:
|
|
return getActivity().getString(R.string.invalid_location);
|
|
case PackageManager.MOVE_FAILED_SYSTEM_PACKAGE:
|
|
return getActivity().getString(R.string.system_package);
|
|
case PackageManager.MOVE_FAILED_INTERNAL_ERROR:
|
|
return "";
|
|
}
|
|
return "";
|
|
}
|
|
|
|
private void initMoveButton() {
|
|
if (Environment.isExternalStorageEmulated()) {
|
|
mMoveAppButton.setVisibility(View.INVISIBLE);
|
|
return;
|
|
}
|
|
boolean dataOnly = false;
|
|
dataOnly = (mPackageInfo == null) && (mAppEntry != null);
|
|
boolean moveDisable = true;
|
|
if (dataOnly) {
|
|
mMoveAppButton.setText(R.string.move_app);
|
|
} else if ((mAppEntry.info.flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE) != 0) {
|
|
mMoveAppButton.setText(R.string.move_app_to_internal);
|
|
// Always let apps move to internal storage from sdcard.
|
|
moveDisable = false;
|
|
} else {
|
|
mMoveAppButton.setText(R.string.move_app_to_sdcard);
|
|
mCanBeOnSdCardChecker.init();
|
|
moveDisable = !mCanBeOnSdCardChecker.check(mAppEntry.info);
|
|
}
|
|
if (moveDisable) {
|
|
mMoveAppButton.setEnabled(false);
|
|
} else {
|
|
mMoveAppButton.setOnClickListener(this);
|
|
mMoveAppButton.setEnabled(true);
|
|
}
|
|
}
|
|
|
|
private boolean isThisASystemPackage() {
|
|
try {
|
|
PackageInfo sys = mPm.getPackageInfo("android", PackageManager.GET_SIGNATURES);
|
|
return (mPackageInfo != null && mPackageInfo.signatures != null &&
|
|
sys.signatures[0].equals(mPackageInfo.signatures[0]));
|
|
} catch (PackageManager.NameNotFoundException e) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
private boolean handleDisableable(Button button) {
|
|
boolean disableable = false;
|
|
// Try to prevent the user from bricking their phone
|
|
// by not allowing disabling of apps signed with the
|
|
// system cert and any launcher app in the system.
|
|
if (mHomePackages.contains(mAppEntry.info.packageName) || isThisASystemPackage()) {
|
|
// Disable button for core system applications.
|
|
button.setText(R.string.disable_text);
|
|
} else if (mAppEntry.info.enabled) {
|
|
button.setText(R.string.disable_text);
|
|
disableable = true;
|
|
} else {
|
|
button.setText(R.string.enable_text);
|
|
disableable = true;
|
|
}
|
|
|
|
return disableable;
|
|
}
|
|
|
|
private void initUninstallButtons() {
|
|
mUpdatedSysApp = (mAppEntry.info.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) != 0;
|
|
boolean enabled = true;
|
|
if (mUpdatedSysApp) {
|
|
mUninstallButton.setText(R.string.app_factory_reset);
|
|
boolean specialDisable = false;
|
|
if ((mAppEntry.info.flags & ApplicationInfo.FLAG_SYSTEM) != 0) {
|
|
specialDisable = handleDisableable(mSpecialDisableButton);
|
|
mSpecialDisableButton.setOnClickListener(this);
|
|
}
|
|
mMoreControlButtons.setVisibility(specialDisable ? View.VISIBLE : View.GONE);
|
|
} else {
|
|
mMoreControlButtons.setVisibility(View.GONE);
|
|
if ((mAppEntry.info.flags & ApplicationInfo.FLAG_SYSTEM) != 0) {
|
|
enabled = handleDisableable(mUninstallButton);
|
|
} else if ((mPackageInfo.applicationInfo.flags
|
|
& ApplicationInfo.FLAG_INSTALLED) == 0
|
|
&& mUserManager.getUsers().size() >= 2) {
|
|
// When we have multiple users, there is a separate menu
|
|
// to uninstall for all users.
|
|
mUninstallButton.setText(R.string.uninstall_text);
|
|
enabled = false;
|
|
} else {
|
|
mUninstallButton.setText(R.string.uninstall_text);
|
|
}
|
|
}
|
|
// If this is a device admin, it can't be uninstalled or disabled.
|
|
// We do this here so the text of the button is still set correctly.
|
|
if (mDpm.packageHasActiveAdmins(mPackageInfo.packageName)) {
|
|
enabled = false;
|
|
}
|
|
|
|
// If this is the default (or only) home app, suppress uninstall (even if
|
|
// we still think it should be allowed for other reasons)
|
|
if (enabled && mHomePackages.contains(mPackageInfo.packageName)) {
|
|
ArrayList<ResolveInfo> homeActivities = new ArrayList<ResolveInfo>();
|
|
ComponentName currentDefaultHome = mPm.getHomeActivities(homeActivities);
|
|
if (currentDefaultHome == null) {
|
|
// No preferred default, so permit uninstall only when
|
|
// there is more than one candidate
|
|
enabled = (mHomePackages.size() > 1);
|
|
} else {
|
|
// There is an explicit default home app -- forbid uninstall of
|
|
// that one, but permit it for installed-but-inactive ones.
|
|
enabled = !mPackageInfo.packageName.equals(currentDefaultHome.getPackageName());
|
|
}
|
|
}
|
|
|
|
mUninstallButton.setEnabled(enabled);
|
|
if (enabled) {
|
|
// Register listener
|
|
mUninstallButton.setOnClickListener(this);
|
|
}
|
|
}
|
|
|
|
private void initNotificationButton() {
|
|
INotificationManager nm = INotificationManager.Stub.asInterface(
|
|
ServiceManager.getService(Context.NOTIFICATION_SERVICE));
|
|
boolean enabled = true; // default on
|
|
try {
|
|
enabled = nm.areNotificationsEnabledForPackage(mAppEntry.info.packageName,
|
|
mAppEntry.info.uid);
|
|
} catch (android.os.RemoteException ex) {
|
|
// this does not bode well
|
|
}
|
|
mNotificationSwitch.setChecked(enabled);
|
|
if (isThisASystemPackage()) {
|
|
mNotificationSwitch.setEnabled(false);
|
|
} else {
|
|
mNotificationSwitch.setEnabled(true);
|
|
mNotificationSwitch.setOnCheckedChangeListener(this);
|
|
}
|
|
}
|
|
|
|
/** Called when the activity is first created. */
|
|
@Override
|
|
public void onCreate(Bundle icicle) {
|
|
super.onCreate(icicle);
|
|
|
|
mState = ApplicationsState.getInstance(getActivity().getApplication());
|
|
mSession = mState.newSession(this);
|
|
mPm = getActivity().getPackageManager();
|
|
mUserManager = (UserManager)getActivity().getSystemService(Context.USER_SERVICE);
|
|
IBinder b = ServiceManager.getService(Context.USB_SERVICE);
|
|
mUsbManager = IUsbManager.Stub.asInterface(b);
|
|
mAppWidgetManager = AppWidgetManager.getInstance(getActivity());
|
|
mDpm = (DevicePolicyManager)getActivity().getSystemService(Context.DEVICE_POLICY_SERVICE);
|
|
mSmsManager = ISms.Stub.asInterface(ServiceManager.getService("isms"));
|
|
|
|
mCanBeOnSdCardChecker = new CanBeOnSdCardChecker();
|
|
|
|
// Need to make sure we have loaded applications at this point.
|
|
mSession.resume();
|
|
|
|
retrieveAppEntry();
|
|
|
|
setHasOptionsMenu(true);
|
|
}
|
|
|
|
@Override
|
|
public View onCreateView(
|
|
LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
|
|
final View view = inflater.inflate(R.layout.installed_app_details, container, false);
|
|
Utils.prepareCustomPreferencesList(container, view, view, false);
|
|
|
|
mRootView = view;
|
|
mComputingStr = getActivity().getText(R.string.computing_size);
|
|
|
|
// Set default values on sizes
|
|
mTotalSize = (TextView)view.findViewById(R.id.total_size_text);
|
|
mAppSize = (TextView)view.findViewById(R.id.application_size_text);
|
|
mDataSize = (TextView)view.findViewById(R.id.data_size_text);
|
|
mExternalCodeSize = (TextView)view.findViewById(R.id.external_code_size_text);
|
|
mExternalDataSize = (TextView)view.findViewById(R.id.external_data_size_text);
|
|
|
|
if (Environment.isExternalStorageEmulated()) {
|
|
((View)mExternalCodeSize.getParent()).setVisibility(View.GONE);
|
|
((View)mExternalDataSize.getParent()).setVisibility(View.GONE);
|
|
}
|
|
|
|
// Get Control button panel
|
|
View btnPanel = view.findViewById(R.id.control_buttons_panel);
|
|
mForceStopButton = (Button) btnPanel.findViewById(R.id.left_button);
|
|
mForceStopButton.setText(R.string.force_stop);
|
|
mUninstallButton = (Button)btnPanel.findViewById(R.id.right_button);
|
|
mForceStopButton.setEnabled(false);
|
|
|
|
// Get More Control button panel
|
|
mMoreControlButtons = view.findViewById(R.id.more_control_buttons_panel);
|
|
mMoreControlButtons.findViewById(R.id.left_button).setVisibility(View.INVISIBLE);
|
|
mSpecialDisableButton = (Button)mMoreControlButtons.findViewById(R.id.right_button);
|
|
mMoreControlButtons.setVisibility(View.GONE);
|
|
|
|
// Initialize clear data and move install location buttons
|
|
View data_buttons_panel = view.findViewById(R.id.data_buttons_panel);
|
|
mClearDataButton = (Button) data_buttons_panel.findViewById(R.id.right_button);
|
|
mMoveAppButton = (Button) data_buttons_panel.findViewById(R.id.left_button);
|
|
|
|
// Cache section
|
|
mCacheSize = (TextView) view.findViewById(R.id.cache_size_text);
|
|
mClearCacheButton = (Button) view.findViewById(R.id.clear_cache_button);
|
|
|
|
mActivitiesButton = (Button)view.findViewById(R.id.clear_activities_button);
|
|
|
|
// Screen compatibility control
|
|
mScreenCompatSection = view.findViewById(R.id.screen_compatibility_section);
|
|
mAskCompatibilityCB = (CheckBox)view.findViewById(R.id.ask_compatibility_cb);
|
|
mEnableCompatibilityCB = (CheckBox)view.findViewById(R.id.enable_compatibility_cb);
|
|
|
|
mNotificationSwitch = (CompoundButton) view.findViewById(R.id.notification_switch);
|
|
|
|
return view;
|
|
}
|
|
|
|
@Override
|
|
public void onCreateOptionsMenu(Menu menu, MenuInflater inflater) {
|
|
menu.add(0, UNINSTALL_ALL_USERS_MENU, 1, R.string.uninstall_all_users_text)
|
|
.setShowAsAction(MenuItem.SHOW_AS_ACTION_NEVER);
|
|
}
|
|
|
|
@Override
|
|
public void onPrepareOptionsMenu(Menu menu) {
|
|
boolean showIt = true;
|
|
if (mUpdatedSysApp) {
|
|
showIt = false;
|
|
} else if (mAppEntry == null) {
|
|
showIt = false;
|
|
} else if ((mAppEntry.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;
|
|
}
|
|
menu.findItem(UNINSTALL_ALL_USERS_MENU).setVisible(showIt);
|
|
}
|
|
|
|
@Override
|
|
public boolean onOptionsItemSelected(MenuItem item) {
|
|
int menuId = item.getItemId();
|
|
if (menuId == UNINSTALL_ALL_USERS_MENU) {
|
|
uninstallPkg(mAppEntry.info.packageName, true, false);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public void onActivityResult(int requestCode, int resultCode, Intent data) {
|
|
super.onActivityResult(requestCode, resultCode, data);
|
|
if (requestCode == REQUEST_UNINSTALL) {
|
|
if (mDisableAfterUninstall) {
|
|
mDisableAfterUninstall = false;
|
|
try {
|
|
ApplicationInfo ainfo = getActivity().getPackageManager().getApplicationInfo(
|
|
mAppEntry.info.packageName, PackageManager.GET_UNINSTALLED_PACKAGES
|
|
| PackageManager.GET_DISABLED_COMPONENTS);
|
|
if ((ainfo.flags & ApplicationInfo.FLAG_UPDATED_SYSTEM_APP) == 0) {
|
|
new DisableChanger(this, mAppEntry.info,
|
|
PackageManager.COMPONENT_ENABLED_STATE_DISABLED_USER)
|
|
.execute((Object)null);
|
|
}
|
|
} catch (NameNotFoundException e) {
|
|
}
|
|
}
|
|
if (!refreshUi()) {
|
|
setIntentAndFinish(true, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Utility method to set application label and icon.
|
|
private void setAppLabelAndIcon(PackageInfo pkgInfo) {
|
|
final View appSnippet = mRootView.findViewById(R.id.app_snippet);
|
|
appSnippet.setPaddingRelative(0, appSnippet.getPaddingTop(), 0, appSnippet.getPaddingBottom());
|
|
|
|
ImageView icon = (ImageView) appSnippet.findViewById(R.id.app_icon);
|
|
mState.ensureIcon(mAppEntry);
|
|
icon.setImageDrawable(mAppEntry.icon);
|
|
// Set application name.
|
|
TextView label = (TextView) appSnippet.findViewById(R.id.app_name);
|
|
label.setText(mAppEntry.label);
|
|
// Version number of application
|
|
mAppVersion = (TextView) appSnippet.findViewById(R.id.app_size);
|
|
|
|
if (pkgInfo != null && pkgInfo.versionName != null) {
|
|
mAppVersion.setVisibility(View.VISIBLE);
|
|
mAppVersion.setText(getActivity().getString(R.string.version_text,
|
|
String.valueOf(pkgInfo.versionName)));
|
|
} else {
|
|
mAppVersion.setVisibility(View.INVISIBLE);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onResume() {
|
|
super.onResume();
|
|
|
|
mSession.resume();
|
|
if (!refreshUi()) {
|
|
setIntentAndFinish(true, true);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onPause() {
|
|
super.onPause();
|
|
mSession.pause();
|
|
}
|
|
|
|
@Override
|
|
public void onDestroyView() {
|
|
super.onDestroyView();
|
|
mSession.release();
|
|
}
|
|
|
|
@Override
|
|
public void onAllSizesComputed() {
|
|
}
|
|
|
|
@Override
|
|
public void onPackageIconChanged() {
|
|
}
|
|
|
|
@Override
|
|
public void onPackageListChanged() {
|
|
refreshUi();
|
|
}
|
|
|
|
@Override
|
|
public void onRebuildComplete(ArrayList<AppEntry> apps) {
|
|
}
|
|
|
|
@Override
|
|
public void onPackageSizeChanged(String packageName) {
|
|
if (packageName.equals(mAppEntry.info.packageName)) {
|
|
refreshSizeInfo();
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onRunningStateChanged(boolean running) {
|
|
}
|
|
|
|
private String retrieveAppEntry() {
|
|
final Bundle args = getArguments();
|
|
String packageName = (args != null) ? args.getString(ARG_PACKAGE_NAME) : null;
|
|
if (packageName == null) {
|
|
Intent intent = (args == null) ?
|
|
getActivity().getIntent() : (Intent) args.getParcelable("intent");
|
|
if (intent != null) {
|
|
packageName = intent.getData().getSchemeSpecificPart();
|
|
}
|
|
}
|
|
mAppEntry = mState.getEntry(packageName);
|
|
if (mAppEntry != null) {
|
|
// Get application info again to refresh changed properties of application
|
|
try {
|
|
mPackageInfo = mPm.getPackageInfo(mAppEntry.info.packageName,
|
|
PackageManager.GET_DISABLED_COMPONENTS |
|
|
PackageManager.GET_UNINSTALLED_PACKAGES |
|
|
PackageManager.GET_SIGNATURES);
|
|
} catch (NameNotFoundException e) {
|
|
Log.e(TAG, "Exception when retrieving package:" + mAppEntry.info.packageName, e);
|
|
}
|
|
} else {
|
|
Log.w(TAG, "Missing AppEntry; maybe reinstalling?");
|
|
mPackageInfo = null;
|
|
}
|
|
|
|
return packageName;
|
|
}
|
|
|
|
private boolean signaturesMatch(String pkg1, String pkg2) {
|
|
if (pkg1 != null && pkg2 != null) {
|
|
try {
|
|
final int match = mPm.checkSignatures(pkg1, pkg2);
|
|
if (match >= PackageManager.SIGNATURE_MATCH) {
|
|
return true;
|
|
}
|
|
} catch (Exception e) {
|
|
// e.g. named alternate package not found during lookup;
|
|
// this is an expected case sometimes
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
private boolean refreshUi() {
|
|
if (mMoveInProgress) {
|
|
return true;
|
|
}
|
|
final String packageName = 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
|
|
}
|
|
|
|
// Get list of "home" apps and trace through any meta-data references
|
|
List<ResolveInfo> homeActivities = new ArrayList<ResolveInfo>();
|
|
mPm.getHomeActivities(homeActivities);
|
|
mHomePackages.clear();
|
|
for (int i = 0; i< homeActivities.size(); i++) {
|
|
ResolveInfo ri = homeActivities.get(i);
|
|
final String activityPkg = ri.activityInfo.packageName;
|
|
mHomePackages.add(activityPkg);
|
|
|
|
// Also make sure to include anything proxying for the home app
|
|
final Bundle metadata = ri.activityInfo.metaData;
|
|
if (metadata != null) {
|
|
final String metaPkg = metadata.getString(ActivityManager.META_HOME_ALTERNATE);
|
|
if (signaturesMatch(metaPkg, activityPkg)) {
|
|
mHomePackages.add(metaPkg);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Get list of preferred activities
|
|
List<ComponentName> prefActList = new ArrayList<ComponentName>();
|
|
|
|
// Intent list cannot be null. so pass empty list
|
|
List<IntentFilter> intentList = new ArrayList<IntentFilter>();
|
|
mPm.getPreferredActivities(intentList, prefActList, packageName);
|
|
if (localLOGV)
|
|
Log.i(TAG, "Have " + prefActList.size() + " number of activities in preferred list");
|
|
boolean hasUsbDefaults = false;
|
|
try {
|
|
hasUsbDefaults = mUsbManager.hasDefaults(packageName, UserHandle.myUserId());
|
|
} catch (RemoteException e) {
|
|
Log.e(TAG, "mUsbManager.hasDefaults", e);
|
|
}
|
|
boolean hasBindAppWidgetPermission =
|
|
mAppWidgetManager.hasBindAppWidgetPermission(mAppEntry.info.packageName);
|
|
|
|
TextView autoLaunchTitleView = (TextView) mRootView.findViewById(R.id.auto_launch_title);
|
|
TextView autoLaunchView = (TextView) mRootView.findViewById(R.id.auto_launch);
|
|
boolean autoLaunchEnabled = prefActList.size() > 0 || hasUsbDefaults;
|
|
if (!autoLaunchEnabled && !hasBindAppWidgetPermission) {
|
|
resetLaunchDefaultsUi(autoLaunchTitleView, autoLaunchView);
|
|
} else {
|
|
boolean useBullets = hasBindAppWidgetPermission && autoLaunchEnabled;
|
|
|
|
if (hasBindAppWidgetPermission) {
|
|
autoLaunchTitleView.setText(R.string.auto_launch_label_generic);
|
|
} else {
|
|
autoLaunchTitleView.setText(R.string.auto_launch_label);
|
|
}
|
|
|
|
CharSequence text = null;
|
|
int bulletIndent = getResources()
|
|
.getDimensionPixelSize(R.dimen.installed_app_details_bullet_offset);
|
|
if (autoLaunchEnabled) {
|
|
CharSequence autoLaunchEnableText = getText(R.string.auto_launch_enable_text);
|
|
SpannableString s = new SpannableString(autoLaunchEnableText);
|
|
if (useBullets) {
|
|
s.setSpan(new BulletSpan(bulletIndent), 0, autoLaunchEnableText.length(), 0);
|
|
}
|
|
text = (text == null) ?
|
|
TextUtils.concat(s, "\n") : TextUtils.concat(text, "\n", s, "\n");
|
|
}
|
|
if (hasBindAppWidgetPermission) {
|
|
CharSequence alwaysAllowBindAppWidgetsText =
|
|
getText(R.string.always_allow_bind_appwidgets_text);
|
|
SpannableString s = new SpannableString(alwaysAllowBindAppWidgetsText);
|
|
if (useBullets) {
|
|
s.setSpan(new BulletSpan(bulletIndent),
|
|
0, alwaysAllowBindAppWidgetsText.length(), 0);
|
|
}
|
|
text = (text == null) ?
|
|
TextUtils.concat(s, "\n") : TextUtils.concat(text, "\n", s, "\n");
|
|
}
|
|
autoLaunchView.setText(text);
|
|
mActivitiesButton.setEnabled(true);
|
|
mActivitiesButton.setOnClickListener(this);
|
|
}
|
|
|
|
// Screen compatibility section.
|
|
ActivityManager am = (ActivityManager)
|
|
getActivity().getSystemService(Context.ACTIVITY_SERVICE);
|
|
int compatMode = am.getPackageScreenCompatMode(packageName);
|
|
// For now these are always off; this is the old UI model which we
|
|
// are no longer using.
|
|
if (false && (compatMode == ActivityManager.COMPAT_MODE_DISABLED
|
|
|| compatMode == ActivityManager.COMPAT_MODE_ENABLED)) {
|
|
mScreenCompatSection.setVisibility(View.VISIBLE);
|
|
mAskCompatibilityCB.setChecked(am.getPackageAskScreenCompat(packageName));
|
|
mAskCompatibilityCB.setOnCheckedChangeListener(this);
|
|
mEnableCompatibilityCB.setChecked(compatMode == ActivityManager.COMPAT_MODE_ENABLED);
|
|
mEnableCompatibilityCB.setOnCheckedChangeListener(this);
|
|
} else {
|
|
mScreenCompatSection.setVisibility(View.GONE);
|
|
}
|
|
|
|
// Security permissions section
|
|
LinearLayout permsView = (LinearLayout) mRootView.findViewById(R.id.permissions_section);
|
|
AppSecurityPermissions asp = new AppSecurityPermissions(getActivity(), packageName);
|
|
int premiumSmsPermission = getPremiumSmsPermission(packageName);
|
|
// Premium SMS permission implies the app also has SEND_SMS permission, so the original
|
|
// application permissions list doesn't have to be shown/hidden separately. The premium
|
|
// SMS subsection should only be visible if the app has tried to send to a premium SMS.
|
|
if (asp.getPermissionCount() > 0
|
|
|| premiumSmsPermission != SmsUsageMonitor.PREMIUM_SMS_PERMISSION_UNKNOWN) {
|
|
permsView.setVisibility(View.VISIBLE);
|
|
} else {
|
|
permsView.setVisibility(View.GONE);
|
|
}
|
|
// Premium SMS permission subsection
|
|
TextView securityBillingDesc = (TextView) permsView.findViewById(
|
|
R.id.security_settings_billing_desc);
|
|
LinearLayout securityBillingList = (LinearLayout) permsView.findViewById(
|
|
R.id.security_settings_billing_list);
|
|
if (premiumSmsPermission != SmsUsageMonitor.PREMIUM_SMS_PERMISSION_UNKNOWN) {
|
|
// Show the premium SMS permission selector
|
|
securityBillingDesc.setVisibility(View.VISIBLE);
|
|
securityBillingList.setVisibility(View.VISIBLE);
|
|
Spinner spinner = (Spinner) permsView.findViewById(
|
|
R.id.security_settings_premium_sms_list);
|
|
ArrayAdapter<CharSequence> adapter = ArrayAdapter.createFromResource(getActivity(),
|
|
R.array.security_settings_premium_sms_values,
|
|
android.R.layout.simple_spinner_item);
|
|
adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
|
|
spinner.setAdapter(adapter);
|
|
// List items are in the same order as SmsUsageMonitor constants, offset by 1.
|
|
spinner.setSelection(premiumSmsPermission - 1);
|
|
spinner.setOnItemSelectedListener(new PremiumSmsSelectionListener(
|
|
packageName, mSmsManager));
|
|
} else {
|
|
// Hide the premium SMS permission selector
|
|
securityBillingDesc.setVisibility(View.GONE);
|
|
securityBillingList.setVisibility(View.GONE);
|
|
}
|
|
// App permissions subsection
|
|
if (asp.getPermissionCount() > 0) {
|
|
// Make the security sections header visible
|
|
LinearLayout securityList = (LinearLayout) permsView.findViewById(
|
|
R.id.security_settings_list);
|
|
securityList.removeAllViews();
|
|
securityList.addView(asp.getPermissionsViewWithRevokeButtons());
|
|
// If this app is running under a shared user ID with other apps,
|
|
// update the description to explain this.
|
|
String[] packages = mPm.getPackagesForUid(mPackageInfo.applicationInfo.uid);
|
|
if (packages != null && packages.length > 1) {
|
|
ArrayList<CharSequence> pnames = new ArrayList<CharSequence>();
|
|
for (int i=0; i<packages.length; i++) {
|
|
String pkg = packages[i];
|
|
if (mPackageInfo.packageName.equals(pkg)) {
|
|
continue;
|
|
}
|
|
try {
|
|
ApplicationInfo ainfo = mPm.getApplicationInfo(pkg, 0);
|
|
pnames.add(ainfo.loadLabel(mPm));
|
|
} catch (PackageManager.NameNotFoundException e) {
|
|
}
|
|
}
|
|
final int N = pnames.size();
|
|
if (N > 0) {
|
|
final Resources res = getActivity().getResources();
|
|
String appListStr;
|
|
if (N == 1) {
|
|
appListStr = pnames.get(0).toString();
|
|
} else if (N == 2) {
|
|
appListStr = res.getString(R.string.join_two_items, pnames.get(0),
|
|
pnames.get(1));
|
|
} else {
|
|
appListStr = pnames.get(N-2).toString();
|
|
for (int i=N-3; i>=0; i--) {
|
|
appListStr = res.getString(i == 0 ? R.string.join_many_items_first
|
|
: R.string.join_many_items_middle, pnames.get(i), appListStr);
|
|
}
|
|
appListStr = res.getString(R.string.join_many_items_last,
|
|
appListStr, pnames.get(N-1));
|
|
}
|
|
TextView descr = (TextView) mRootView.findViewById(
|
|
R.id.security_settings_desc);
|
|
descr.setText(res.getString(R.string.security_settings_desc_multi,
|
|
mPackageInfo.applicationInfo.loadLabel(mPm), appListStr));
|
|
}
|
|
}
|
|
}
|
|
|
|
checkForceStop();
|
|
setAppLabelAndIcon(mPackageInfo);
|
|
refreshButtons();
|
|
refreshSizeInfo();
|
|
|
|
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 {
|
|
ApplicationInfo ainfo = getActivity().getPackageManager().getApplicationInfo(
|
|
mAppEntry.info.packageName, PackageManager.GET_UNINSTALLED_PACKAGES
|
|
| PackageManager.GET_DISABLED_COMPONENTS);
|
|
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;
|
|
}
|
|
|
|
private static class PremiumSmsSelectionListener implements AdapterView.OnItemSelectedListener {
|
|
private final String mPackageName;
|
|
private final ISms mSmsManager;
|
|
|
|
PremiumSmsSelectionListener(String packageName, ISms smsManager) {
|
|
mPackageName = packageName;
|
|
mSmsManager = smsManager;
|
|
}
|
|
|
|
@Override
|
|
public void onItemSelected(AdapterView<?> parent, View view, int position,
|
|
long id) {
|
|
if (position >= 0 && position < 3) {
|
|
Log.d(TAG, "Selected premium SMS policy " + position);
|
|
setPremiumSmsPermission(mPackageName, (position + 1));
|
|
} else {
|
|
Log.e(TAG, "Error: unknown premium SMS policy " + position);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onNothingSelected(AdapterView<?> parent) {
|
|
// Ignored
|
|
}
|
|
|
|
private void setPremiumSmsPermission(String packageName, int permission) {
|
|
try {
|
|
if (mSmsManager != null) {
|
|
mSmsManager.setPremiumSmsPermission(packageName, permission);
|
|
}
|
|
} catch (RemoteException ex) {
|
|
// ignored
|
|
}
|
|
}
|
|
}
|
|
|
|
private void resetLaunchDefaultsUi(TextView title, TextView autoLaunchView) {
|
|
title.setText(R.string.auto_launch_label);
|
|
autoLaunchView.setText(R.string.auto_launch_disable_text);
|
|
// Disable clear activities button
|
|
mActivitiesButton.setEnabled(false);
|
|
}
|
|
|
|
private void setIntentAndFinish(boolean finish, boolean appChanged) {
|
|
if(localLOGV) Log.i(TAG, "appChanged="+appChanged);
|
|
Intent intent = new Intent();
|
|
intent.putExtra(ManageApplications.APP_CHG, appChanged);
|
|
SettingsActivity sa = (SettingsActivity)getActivity();
|
|
sa.finishPreferencePanel(this, Activity.RESULT_OK, intent);
|
|
}
|
|
|
|
private void refreshSizeInfo() {
|
|
if (mAppEntry.size == ApplicationsState.SIZE_INVALID
|
|
|| mAppEntry.size == ApplicationsState.SIZE_UNKNOWN) {
|
|
mLastCodeSize = mLastDataSize = mLastCacheSize = mLastTotalSize = -1;
|
|
if (!mHaveSizes) {
|
|
mAppSize.setText(mComputingStr);
|
|
mDataSize.setText(mComputingStr);
|
|
mCacheSize.setText(mComputingStr);
|
|
mTotalSize.setText(mComputingStr);
|
|
}
|
|
mClearDataButton.setEnabled(false);
|
|
mClearCacheButton.setEnabled(false);
|
|
|
|
} else {
|
|
mHaveSizes = true;
|
|
long codeSize = mAppEntry.codeSize;
|
|
long dataSize = mAppEntry.dataSize;
|
|
if (Environment.isExternalStorageEmulated()) {
|
|
codeSize += mAppEntry.externalCodeSize;
|
|
dataSize += mAppEntry.externalDataSize;
|
|
} else {
|
|
if (mLastExternalCodeSize != mAppEntry.externalCodeSize) {
|
|
mLastExternalCodeSize = mAppEntry.externalCodeSize;
|
|
mExternalCodeSize.setText(getSizeStr(mAppEntry.externalCodeSize));
|
|
}
|
|
if (mLastExternalDataSize != mAppEntry.externalDataSize) {
|
|
mLastExternalDataSize = mAppEntry.externalDataSize;
|
|
mExternalDataSize.setText(getSizeStr( mAppEntry.externalDataSize));
|
|
}
|
|
}
|
|
if (mLastCodeSize != codeSize) {
|
|
mLastCodeSize = codeSize;
|
|
mAppSize.setText(getSizeStr(codeSize));
|
|
}
|
|
if (mLastDataSize != dataSize) {
|
|
mLastDataSize = dataSize;
|
|
mDataSize.setText(getSizeStr(dataSize));
|
|
}
|
|
long cacheSize = mAppEntry.cacheSize + mAppEntry.externalCacheSize;
|
|
if (mLastCacheSize != cacheSize) {
|
|
mLastCacheSize = cacheSize;
|
|
mCacheSize.setText(getSizeStr(cacheSize));
|
|
}
|
|
if (mLastTotalSize != mAppEntry.size) {
|
|
mLastTotalSize = mAppEntry.size;
|
|
mTotalSize.setText(getSizeStr(mAppEntry.size));
|
|
}
|
|
|
|
if ((mAppEntry.dataSize+ mAppEntry.externalDataSize) <= 0 || !mCanClearData) {
|
|
mClearDataButton.setEnabled(false);
|
|
} else {
|
|
mClearDataButton.setEnabled(true);
|
|
mClearDataButton.setOnClickListener(this);
|
|
}
|
|
if (cacheSize <= 0) {
|
|
mClearCacheButton.setEnabled(false);
|
|
} else {
|
|
mClearCacheButton.setEnabled(true);
|
|
mClearCacheButton.setOnClickListener(this);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Private method to handle clear message notification from observer when
|
|
* the async operation from PackageManager is complete
|
|
*/
|
|
private void processClearMsg(Message msg) {
|
|
int result = msg.arg1;
|
|
String packageName = mAppEntry.info.packageName;
|
|
mClearDataButton.setText(R.string.clear_user_data_text);
|
|
if(result == OP_SUCCESSFUL) {
|
|
Log.i(TAG, "Cleared user data for package : "+packageName);
|
|
mState.requestSize(mAppEntry.info.packageName);
|
|
} else {
|
|
mClearDataButton.setEnabled(true);
|
|
}
|
|
checkForceStop();
|
|
}
|
|
|
|
private void refreshButtons() {
|
|
if (!mMoveInProgress) {
|
|
initUninstallButtons();
|
|
initDataButtons();
|
|
initMoveButton();
|
|
initNotificationButton();
|
|
} else {
|
|
mMoveAppButton.setText(R.string.moving);
|
|
mMoveAppButton.setEnabled(false);
|
|
mUninstallButton.setEnabled(false);
|
|
mSpecialDisableButton.setEnabled(false);
|
|
}
|
|
}
|
|
|
|
private void processMoveMsg(Message msg) {
|
|
int result = msg.arg1;
|
|
String packageName = mAppEntry.info.packageName;
|
|
// Refresh the button attributes.
|
|
mMoveInProgress = false;
|
|
if (result == PackageManager.MOVE_SUCCEEDED) {
|
|
Log.i(TAG, "Moved resources for " + packageName);
|
|
// Refresh size information again.
|
|
mState.requestSize(mAppEntry.info.packageName);
|
|
} else {
|
|
showDialogInner(DLG_MOVE_FAILED, result);
|
|
}
|
|
refreshUi();
|
|
}
|
|
|
|
/*
|
|
* Private method to initiate clearing user data when the user clicks the clear data
|
|
* button for a system package
|
|
*/
|
|
private void initiateClearUserData() {
|
|
mClearDataButton.setEnabled(false);
|
|
// Invoke uninstall or clear user data based on sysPackage
|
|
String packageName = mAppEntry.info.packageName;
|
|
Log.i(TAG, "Clearing user data for package : " + packageName);
|
|
if (mClearDataObserver == null) {
|
|
mClearDataObserver = new ClearUserDataObserver();
|
|
}
|
|
ActivityManager am = (ActivityManager)
|
|
getActivity().getSystemService(Context.ACTIVITY_SERVICE);
|
|
boolean res = am.clearApplicationUserData(packageName, mClearDataObserver);
|
|
if (!res) {
|
|
// Clearing data failed for some obscure reason. Just log error for now
|
|
Log.i(TAG, "Couldnt clear application user data for package:"+packageName);
|
|
showDialogInner(DLG_CANNOT_CLEAR_DATA, 0);
|
|
} else {
|
|
mClearDataButton.setText(R.string.recompute_size);
|
|
}
|
|
}
|
|
|
|
private void showDialogInner(int id, int moveErrorCode) {
|
|
DialogFragment newFragment = MyAlertDialogFragment.newInstance(id, moveErrorCode);
|
|
newFragment.setTargetFragment(this, 0);
|
|
newFragment.show(getFragmentManager(), "dialog " + id);
|
|
}
|
|
|
|
public static class MyAlertDialogFragment extends DialogFragment {
|
|
|
|
public static MyAlertDialogFragment newInstance(int id, int moveErrorCode) {
|
|
MyAlertDialogFragment frag = new MyAlertDialogFragment();
|
|
Bundle args = new Bundle();
|
|
args.putInt("id", id);
|
|
args.putInt("moveError", moveErrorCode);
|
|
frag.setArguments(args);
|
|
return frag;
|
|
}
|
|
|
|
InstalledAppDetails getOwner() {
|
|
return (InstalledAppDetails)getTargetFragment();
|
|
}
|
|
|
|
@Override
|
|
public Dialog onCreateDialog(Bundle savedInstanceState) {
|
|
int id = getArguments().getInt("id");
|
|
int moveErrorCode = getArguments().getInt("moveError");
|
|
switch (id) {
|
|
case DLG_CLEAR_DATA:
|
|
return new AlertDialog.Builder(getActivity())
|
|
.setTitle(getActivity().getText(R.string.clear_data_dlg_title))
|
|
.setIconAttribute(android.R.attr.alertDialogIcon)
|
|
.setMessage(getActivity().getText(R.string.clear_data_dlg_text))
|
|
.setPositiveButton(R.string.dlg_ok,
|
|
new DialogInterface.OnClickListener() {
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
// Clear user data here
|
|
getOwner().initiateClearUserData();
|
|
}
|
|
})
|
|
.setNegativeButton(R.string.dlg_cancel, null)
|
|
.create();
|
|
case DLG_FACTORY_RESET:
|
|
return new AlertDialog.Builder(getActivity())
|
|
.setTitle(getActivity().getText(R.string.app_factory_reset_dlg_title))
|
|
.setIconAttribute(android.R.attr.alertDialogIcon)
|
|
.setMessage(getActivity().getText(R.string.app_factory_reset_dlg_text))
|
|
.setPositiveButton(R.string.dlg_ok,
|
|
new DialogInterface.OnClickListener() {
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
// Clear user data here
|
|
getOwner().uninstallPkg(getOwner().mAppEntry.info.packageName,
|
|
false, false);
|
|
}
|
|
})
|
|
.setNegativeButton(R.string.dlg_cancel, null)
|
|
.create();
|
|
case DLG_APP_NOT_FOUND:
|
|
return new AlertDialog.Builder(getActivity())
|
|
.setTitle(getActivity().getText(R.string.app_not_found_dlg_title))
|
|
.setIconAttribute(android.R.attr.alertDialogIcon)
|
|
.setMessage(getActivity().getText(R.string.app_not_found_dlg_title))
|
|
.setNeutralButton(getActivity().getText(R.string.dlg_ok),
|
|
new DialogInterface.OnClickListener() {
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
//force to recompute changed value
|
|
getOwner().setIntentAndFinish(true, true);
|
|
}
|
|
})
|
|
.create();
|
|
case DLG_CANNOT_CLEAR_DATA:
|
|
return new AlertDialog.Builder(getActivity())
|
|
.setTitle(getActivity().getText(R.string.clear_failed_dlg_title))
|
|
.setIconAttribute(android.R.attr.alertDialogIcon)
|
|
.setMessage(getActivity().getText(R.string.clear_failed_dlg_text))
|
|
.setNeutralButton(R.string.dlg_ok,
|
|
new DialogInterface.OnClickListener() {
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
getOwner().mClearDataButton.setEnabled(false);
|
|
//force to recompute changed value
|
|
getOwner().setIntentAndFinish(false, false);
|
|
}
|
|
})
|
|
.create();
|
|
case DLG_FORCE_STOP:
|
|
return new AlertDialog.Builder(getActivity())
|
|
.setTitle(getActivity().getText(R.string.force_stop_dlg_title))
|
|
.setIconAttribute(android.R.attr.alertDialogIcon)
|
|
.setMessage(getActivity().getText(R.string.force_stop_dlg_text))
|
|
.setPositiveButton(R.string.dlg_ok,
|
|
new DialogInterface.OnClickListener() {
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
// Force stop
|
|
getOwner().forceStopPackage(getOwner().mAppEntry.info.packageName);
|
|
}
|
|
})
|
|
.setNegativeButton(R.string.dlg_cancel, null)
|
|
.create();
|
|
case DLG_MOVE_FAILED:
|
|
CharSequence msg = getActivity().getString(R.string.move_app_failed_dlg_text,
|
|
getOwner().getMoveErrMsg(moveErrorCode));
|
|
return new AlertDialog.Builder(getActivity())
|
|
.setTitle(getActivity().getText(R.string.move_app_failed_dlg_title))
|
|
.setIconAttribute(android.R.attr.alertDialogIcon)
|
|
.setMessage(msg)
|
|
.setNeutralButton(R.string.dlg_ok, null)
|
|
.create();
|
|
case DLG_DISABLE:
|
|
return new AlertDialog.Builder(getActivity())
|
|
.setTitle(getActivity().getText(R.string.app_disable_dlg_title))
|
|
.setIconAttribute(android.R.attr.alertDialogIcon)
|
|
.setMessage(getActivity().getText(R.string.app_disable_dlg_text))
|
|
.setPositiveButton(R.string.dlg_ok,
|
|
new DialogInterface.OnClickListener() {
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
// Disable the app
|
|
new DisableChanger(getOwner(), getOwner().mAppEntry.info,
|
|
PackageManager.COMPONENT_ENABLED_STATE_DISABLED_USER)
|
|
.execute((Object)null);
|
|
}
|
|
})
|
|
.setNegativeButton(R.string.dlg_cancel, null)
|
|
.create();
|
|
case DLG_DISABLE_NOTIFICATIONS:
|
|
return new AlertDialog.Builder(getActivity())
|
|
.setTitle(getActivity().getText(R.string.app_disable_notifications_dlg_title))
|
|
.setIconAttribute(android.R.attr.alertDialogIcon)
|
|
.setMessage(getActivity().getText(R.string.app_disable_notifications_dlg_text))
|
|
.setPositiveButton(R.string.dlg_ok,
|
|
new DialogInterface.OnClickListener() {
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
// Disable the package's notifications
|
|
getOwner().setNotificationsEnabled(false);
|
|
}
|
|
})
|
|
.setNegativeButton(R.string.dlg_cancel,
|
|
new DialogInterface.OnClickListener() {
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
// Re-enable the checkbox
|
|
getOwner().mNotificationSwitch.setChecked(true);
|
|
}
|
|
})
|
|
.create();
|
|
case DLG_SPECIAL_DISABLE:
|
|
return new AlertDialog.Builder(getActivity())
|
|
.setTitle(getActivity().getText(R.string.app_special_disable_dlg_title))
|
|
.setIconAttribute(android.R.attr.alertDialogIcon)
|
|
.setMessage(getActivity().getText(R.string.app_special_disable_dlg_text))
|
|
.setPositiveButton(R.string.dlg_ok,
|
|
new DialogInterface.OnClickListener() {
|
|
public void onClick(DialogInterface dialog, int which) {
|
|
// Clear user data here
|
|
getOwner().uninstallPkg(getOwner().mAppEntry.info.packageName,
|
|
false, true);
|
|
}
|
|
})
|
|
.setNegativeButton(R.string.dlg_cancel, null)
|
|
.create();
|
|
}
|
|
throw new IllegalArgumentException("unknown id " + id);
|
|
}
|
|
}
|
|
|
|
private void uninstallPkg(String packageName, boolean allUsers, boolean andDisable) {
|
|
// Create new intent to launch Uninstaller activity
|
|
Uri packageURI = Uri.parse("package:"+packageName);
|
|
Intent uninstallIntent = new Intent(Intent.ACTION_UNINSTALL_PACKAGE, packageURI);
|
|
uninstallIntent.putExtra(Intent.EXTRA_UNINSTALL_ALL_USERS, allUsers);
|
|
startActivityForResult(uninstallIntent, REQUEST_UNINSTALL);
|
|
mDisableAfterUninstall = andDisable;
|
|
}
|
|
|
|
private void forceStopPackage(String pkgName) {
|
|
ActivityManager am = (ActivityManager)getActivity().getSystemService(
|
|
Context.ACTIVITY_SERVICE);
|
|
am.forceStopPackage(pkgName);
|
|
mState.invalidatePackage(pkgName);
|
|
ApplicationsState.AppEntry newEnt = mState.getEntry(pkgName);
|
|
if (newEnt != null) {
|
|
mAppEntry = newEnt;
|
|
}
|
|
checkForceStop();
|
|
}
|
|
|
|
private final BroadcastReceiver mCheckKillProcessesReceiver = new BroadcastReceiver() {
|
|
@Override
|
|
public void onReceive(Context context, Intent intent) {
|
|
updateForceStopButton(getResultCode() != Activity.RESULT_CANCELED);
|
|
}
|
|
};
|
|
|
|
private void updateForceStopButton(boolean enabled) {
|
|
mForceStopButton.setEnabled(enabled);
|
|
mForceStopButton.setOnClickListener(InstalledAppDetails.this);
|
|
}
|
|
|
|
private void checkForceStop() {
|
|
if (mDpm.packageHasActiveAdmins(mPackageInfo.packageName)) {
|
|
// User can't force stop device admin.
|
|
updateForceStopButton(false);
|
|
} else if ((mAppEntry.info.flags&ApplicationInfo.FLAG_STOPPED) == 0) {
|
|
// If the app isn't explicitly stopped, then always show the
|
|
// force stop button.
|
|
updateForceStopButton(true);
|
|
} else {
|
|
Intent intent = new Intent(Intent.ACTION_QUERY_PACKAGE_RESTART,
|
|
Uri.fromParts("package", mAppEntry.info.packageName, null));
|
|
intent.putExtra(Intent.EXTRA_PACKAGES, new String[] { mAppEntry.info.packageName });
|
|
intent.putExtra(Intent.EXTRA_UID, mAppEntry.info.uid);
|
|
intent.putExtra(Intent.EXTRA_USER_HANDLE, UserHandle.getUserId(mAppEntry.info.uid));
|
|
getActivity().sendOrderedBroadcastAsUser(intent, UserHandle.CURRENT, null,
|
|
mCheckKillProcessesReceiver, null, Activity.RESULT_CANCELED, null, null);
|
|
}
|
|
}
|
|
|
|
static class DisableChanger extends AsyncTask<Object, Object, Object> {
|
|
final PackageManager mPm;
|
|
final WeakReference<InstalledAppDetails> mActivity;
|
|
final ApplicationInfo mInfo;
|
|
final int mState;
|
|
|
|
DisableChanger(InstalledAppDetails activity, ApplicationInfo info, int state) {
|
|
mPm = activity.mPm;
|
|
mActivity = new WeakReference<InstalledAppDetails>(activity);
|
|
mInfo = info;
|
|
mState = state;
|
|
}
|
|
|
|
@Override
|
|
protected Object doInBackground(Object... params) {
|
|
mPm.setApplicationEnabledSetting(mInfo.packageName, mState, 0);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
private void setNotificationsEnabled(boolean enabled) {
|
|
String packageName = mAppEntry.info.packageName;
|
|
INotificationManager nm = INotificationManager.Stub.asInterface(
|
|
ServiceManager.getService(Context.NOTIFICATION_SERVICE));
|
|
try {
|
|
final boolean enable = mNotificationSwitch.isChecked();
|
|
nm.setNotificationsEnabledForPackage(packageName, mAppEntry.info.uid, enabled);
|
|
} catch (android.os.RemoteException ex) {
|
|
mNotificationSwitch.setChecked(!enabled); // revert
|
|
}
|
|
}
|
|
|
|
private int getPremiumSmsPermission(String packageName) {
|
|
try {
|
|
if (mSmsManager != null) {
|
|
return mSmsManager.getPremiumSmsPermission(packageName);
|
|
}
|
|
} catch (RemoteException ex) {
|
|
// ignored
|
|
}
|
|
return SmsUsageMonitor.PREMIUM_SMS_PERMISSION_UNKNOWN;
|
|
}
|
|
|
|
/*
|
|
* Method implementing functionality of buttons clicked
|
|
* @see android.view.View.OnClickListener#onClick(android.view.View)
|
|
*/
|
|
public void onClick(View v) {
|
|
String packageName = mAppEntry.info.packageName;
|
|
if(v == mUninstallButton) {
|
|
if (mUpdatedSysApp) {
|
|
showDialogInner(DLG_FACTORY_RESET, 0);
|
|
} else {
|
|
if ((mAppEntry.info.flags & ApplicationInfo.FLAG_SYSTEM) != 0) {
|
|
if (mAppEntry.info.enabled) {
|
|
showDialogInner(DLG_DISABLE, 0);
|
|
} else {
|
|
new DisableChanger(this, mAppEntry.info,
|
|
PackageManager.COMPONENT_ENABLED_STATE_DEFAULT)
|
|
.execute((Object)null);
|
|
}
|
|
} else if ((mAppEntry.info.flags & ApplicationInfo.FLAG_INSTALLED) == 0) {
|
|
uninstallPkg(packageName, true, false);
|
|
} else {
|
|
uninstallPkg(packageName, false, false);
|
|
}
|
|
}
|
|
} else if(v == mSpecialDisableButton) {
|
|
showDialogInner(DLG_SPECIAL_DISABLE, 0);
|
|
} else if(v == mActivitiesButton) {
|
|
mPm.clearPackagePreferredActivities(packageName);
|
|
try {
|
|
mUsbManager.clearDefaults(packageName, UserHandle.myUserId());
|
|
} catch (RemoteException e) {
|
|
Log.e(TAG, "mUsbManager.clearDefaults", e);
|
|
}
|
|
mAppWidgetManager.setBindAppWidgetPermission(packageName, false);
|
|
TextView autoLaunchTitleView =
|
|
(TextView) mRootView.findViewById(R.id.auto_launch_title);
|
|
TextView autoLaunchView = (TextView) mRootView.findViewById(R.id.auto_launch);
|
|
resetLaunchDefaultsUi(autoLaunchTitleView, autoLaunchView);
|
|
} else if(v == mClearDataButton) {
|
|
if (mAppEntry.info.manageSpaceActivityName != null) {
|
|
if (!Utils.isMonkeyRunning()) {
|
|
Intent intent = new Intent(Intent.ACTION_DEFAULT);
|
|
intent.setClassName(mAppEntry.info.packageName,
|
|
mAppEntry.info.manageSpaceActivityName);
|
|
startActivityForResult(intent, REQUEST_MANAGE_SPACE);
|
|
}
|
|
} else {
|
|
showDialogInner(DLG_CLEAR_DATA, 0);
|
|
}
|
|
} else if (v == mClearCacheButton) {
|
|
// Lazy initialization of observer
|
|
if (mClearCacheObserver == null) {
|
|
mClearCacheObserver = new ClearCacheObserver();
|
|
}
|
|
mPm.deleteApplicationCacheFiles(packageName, mClearCacheObserver);
|
|
} else if (v == mForceStopButton) {
|
|
showDialogInner(DLG_FORCE_STOP, 0);
|
|
//forceStopPackage(mAppInfo.packageName);
|
|
} else if (v == mMoveAppButton) {
|
|
if (mPackageMoveObserver == null) {
|
|
mPackageMoveObserver = new PackageMoveObserver();
|
|
}
|
|
int moveFlags = (mAppEntry.info.flags & ApplicationInfo.FLAG_EXTERNAL_STORAGE) != 0 ?
|
|
PackageManager.MOVE_INTERNAL : PackageManager.MOVE_EXTERNAL_MEDIA;
|
|
mMoveInProgress = true;
|
|
refreshButtons();
|
|
mPm.movePackage(mAppEntry.info.packageName, mPackageMoveObserver, moveFlags);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
|
|
String packageName = mAppEntry.info.packageName;
|
|
ActivityManager am = (ActivityManager)
|
|
getActivity().getSystemService(Context.ACTIVITY_SERVICE);
|
|
if (buttonView == mAskCompatibilityCB) {
|
|
am.setPackageAskScreenCompat(packageName, isChecked);
|
|
} else if (buttonView == mEnableCompatibilityCB) {
|
|
am.setPackageScreenCompatMode(packageName, isChecked ?
|
|
ActivityManager.COMPAT_MODE_ENABLED : ActivityManager.COMPAT_MODE_DISABLED);
|
|
} else if (buttonView == mNotificationSwitch) {
|
|
if (!isChecked) {
|
|
showDialogInner(DLG_DISABLE_NOTIFICATIONS, 0);
|
|
} else {
|
|
setNotificationsEnabled(true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|