Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 1 | package com.android.launcher3; |
| 2 | |
| 3 | import android.content.ComponentName; |
| 4 | import android.content.Context; |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 5 | import android.content.Intent; |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 6 | import android.content.pm.ApplicationInfo; |
Sunny Goyal | 3e9be43 | 2017-01-05 15:22:41 -0800 | [diff] [blame] | 7 | import android.content.pm.LauncherActivityInfo; |
Sunny Goyal | f2db253 | 2017-03-01 17:27:16 -0800 | [diff] [blame^] | 8 | import android.content.pm.PackageManager; |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 9 | import android.net.Uri; |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 10 | import android.os.Bundle; |
Sunny Goyal | 7c74e4a | 2016-12-15 15:53:17 -0800 | [diff] [blame] | 11 | import android.os.UserHandle; |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 12 | import android.os.UserManager; |
| 13 | import android.util.AttributeSet; |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 14 | import android.widget.Toast; |
Sunny Goyal | aa8ef11 | 2015-06-12 20:04:41 -0700 | [diff] [blame] | 15 | |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 16 | import com.android.launcher3.compat.LauncherAppsCompat; |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 17 | |
| 18 | public class UninstallDropTarget extends ButtonDropTarget { |
| 19 | |
| 20 | public UninstallDropTarget(Context context, AttributeSet attrs) { |
| 21 | this(context, attrs, 0); |
| 22 | } |
| 23 | |
| 24 | public UninstallDropTarget(Context context, AttributeSet attrs, int defStyle) { |
| 25 | super(context, attrs, defStyle); |
| 26 | } |
| 27 | |
| 28 | @Override |
| 29 | protected void onFinishInflate() { |
| 30 | super.onFinishInflate(); |
| 31 | // Get the hover color |
Sunny Goyal | 3a2698b | 2015-04-28 21:36:46 -0700 | [diff] [blame] | 32 | mHoverColor = getResources().getColor(R.color.uninstall_target_hover_tint); |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 33 | |
Sunny Goyal | 40b626b | 2015-06-04 22:40:14 -0700 | [diff] [blame] | 34 | setDrawable(R.drawable.ic_uninstall_launcher); |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 35 | } |
| 36 | |
| 37 | @Override |
Sunny Goyal | aa8ef11 | 2015-06-12 20:04:41 -0700 | [diff] [blame] | 38 | protected boolean supportsDrop(DragSource source, ItemInfo info) { |
Sunny Goyal | 1a70cef | 2015-04-22 11:29:51 -0700 | [diff] [blame] | 39 | return supportsDrop(getContext(), info); |
| 40 | } |
| 41 | |
| 42 | public static boolean supportsDrop(Context context, Object info) { |
Sunny Goyal | a52ecb0 | 2016-12-16 15:04:51 -0800 | [diff] [blame] | 43 | UserManager userManager = (UserManager) context.getSystemService(Context.USER_SERVICE); |
| 44 | Bundle restrictions = userManager.getUserRestrictions(); |
| 45 | if (restrictions.getBoolean(UserManager.DISALLOW_APPS_CONTROL, false) |
| 46 | || restrictions.getBoolean(UserManager.DISALLOW_UNINSTALL_APPS, false)) { |
| 47 | return false; |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 48 | } |
| 49 | |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 50 | return getUninstallTarget(context, info) != null; |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 51 | } |
| 52 | |
| 53 | /** |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 54 | * @return the component name that should be uninstalled or null. |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 55 | */ |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 56 | private static ComponentName getUninstallTarget(Context context, Object item) { |
| 57 | Intent intent = null; |
Sunny Goyal | 7c74e4a | 2016-12-15 15:53:17 -0800 | [diff] [blame] | 58 | UserHandle user = null; |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 59 | if (item instanceof AppInfo) { |
| 60 | AppInfo info = (AppInfo) item; |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 61 | intent = info.intent; |
| 62 | user = info.user; |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 63 | } else if (item instanceof ShortcutInfo) { |
| 64 | ShortcutInfo info = (ShortcutInfo) item; |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 65 | if (info.itemType == LauncherSettings.BaseLauncherColumns.ITEM_TYPE_APPLICATION) { |
| 66 | // Do not use restore/target intent here as we cannot uninstall an app which is |
| 67 | // being installed/restored. |
| 68 | intent = info.intent; |
| 69 | user = info.user; |
| 70 | } |
| 71 | } |
| 72 | if (intent != null) { |
Sunny Goyal | 3e9be43 | 2017-01-05 15:22:41 -0800 | [diff] [blame] | 73 | LauncherActivityInfo info = LauncherAppsCompat.getInstance(context) |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 74 | .resolveActivity(intent, user); |
| 75 | if (info != null |
| 76 | && (info.getApplicationInfo().flags & ApplicationInfo.FLAG_SYSTEM) == 0) { |
| 77 | return info.getComponentName(); |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 78 | } |
| 79 | } |
| 80 | return null; |
| 81 | } |
| 82 | |
| 83 | @Override |
| 84 | public void onDrop(DragObject d) { |
| 85 | // Differ item deletion |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 86 | if (d.dragSource instanceof DropTargetSource) { |
| 87 | ((DropTargetSource) d.dragSource).deferCompleteDropAfterUninstallActivity(); |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 88 | } |
| 89 | super.onDrop(d); |
| 90 | } |
| 91 | |
| 92 | @Override |
Sunny Goyal | 0f76b56 | 2016-12-13 19:37:10 -0800 | [diff] [blame] | 93 | public void completeDrop(final DragObject d) { |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 94 | DropTargetResultCallback callback = d.dragSource instanceof DropTargetResultCallback |
| 95 | ? (DropTargetResultCallback) d.dragSource : null; |
| 96 | startUninstallActivity(mLauncher, d.dragInfo, callback); |
Tony Wickham | 734dfbe | 2015-09-16 16:22:36 -0700 | [diff] [blame] | 97 | } |
| 98 | |
Sunny Goyal | aa8ef11 | 2015-06-12 20:04:41 -0700 | [diff] [blame] | 99 | public static boolean startUninstallActivity(Launcher launcher, ItemInfo info) { |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 100 | return startUninstallActivity(launcher, info, null); |
Sunny Goyal | 1a70cef | 2015-04-22 11:29:51 -0700 | [diff] [blame] | 101 | } |
| 102 | |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 103 | public static boolean startUninstallActivity( |
| 104 | final Launcher launcher, ItemInfo info, DropTargetResultCallback callback) { |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 105 | final ComponentName cn = getUninstallTarget(launcher, info); |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 106 | |
| 107 | final boolean isUninstallable; |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 108 | if (cn == null) { |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 109 | // System applications cannot be installed. For now, show a toast explaining that. |
| 110 | // We may give them the option of disabling apps this way. |
| 111 | Toast.makeText(launcher, R.string.uninstall_system_app_text, Toast.LENGTH_SHORT).show(); |
| 112 | isUninstallable = false; |
| 113 | } else { |
| 114 | Intent intent = new Intent(Intent.ACTION_DELETE, |
| 115 | Uri.fromParts("package", cn.getPackageName(), cn.getClassName())) |
| 116 | .setFlags(Intent.FLAG_ACTIVITY_NEW_TASK |
| 117 | | Intent.FLAG_ACTIVITY_EXCLUDE_FROM_RECENTS); |
Sunny Goyal | 7c74e4a | 2016-12-15 15:53:17 -0800 | [diff] [blame] | 118 | intent.putExtra(Intent.EXTRA_USER, info.user); |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 119 | launcher.startActivity(intent); |
| 120 | isUninstallable = true; |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 121 | } |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 122 | if (callback != null) { |
Sunny Goyal | 8e0e1d7 | 2016-10-10 10:41:41 -0700 | [diff] [blame] | 123 | sendUninstallResult(launcher, isUninstallable, cn, info.user, callback); |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 124 | } |
| 125 | return isUninstallable; |
| 126 | } |
| 127 | |
| 128 | /** |
| 129 | * Notifies the {@param callback} whether the uninstall was successful or not. |
| 130 | * |
| 131 | * Since there is no direct callback for an uninstall request, we check the package existence |
| 132 | * when the launch resumes next time. This assumes that the uninstall activity will finish only |
| 133 | * after the task is completed |
| 134 | */ |
| 135 | protected static void sendUninstallResult( |
| 136 | final Launcher launcher, boolean activityStarted, |
Sunny Goyal | 7c74e4a | 2016-12-15 15:53:17 -0800 | [diff] [blame] | 137 | final ComponentName cn, final UserHandle user, |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 138 | final DropTargetResultCallback callback) { |
| 139 | if (activityStarted) { |
| 140 | final Runnable checkIfUninstallWasSuccess = new Runnable() { |
| 141 | @Override |
| 142 | public void run() { |
Sunny Goyal | f2db253 | 2017-03-01 17:27:16 -0800 | [diff] [blame^] | 143 | // We use MATCH_UNINSTALLED_PACKAGES as the app can be on SD card as well. |
| 144 | boolean uninstallSuccessful = LauncherAppsCompat.getInstance(launcher) |
| 145 | .getApplicationInfo(cn.getPackageName(), |
| 146 | PackageManager.MATCH_UNINSTALLED_PACKAGES, user) == null; |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 147 | callback.onDragObjectRemoved(uninstallSuccessful); |
| 148 | } |
| 149 | }; |
| 150 | launcher.addOnResumeCallback(checkIfUninstallWasSuccess); |
| 151 | } else { |
| 152 | callback.onDragObjectRemoved(false); |
| 153 | } |
| 154 | } |
| 155 | |
| 156 | public interface DropTargetResultCallback { |
| 157 | /** |
| 158 | * A drag operation was complete. |
| 159 | * @param isRemoved true if the drag object should be removed, false otherwise. |
| 160 | */ |
| 161 | void onDragObjectRemoved(boolean isRemoved); |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 162 | } |
| 163 | |
| 164 | /** |
| 165 | * Interface defining an object that can provide uninstallable drag objects. |
| 166 | */ |
Sunny Goyal | d5bd67d | 2016-03-11 01:10:19 -0800 | [diff] [blame] | 167 | public interface DropTargetSource extends DropTargetResultCallback { |
Sunny Goyal | fa401a1 | 2015-04-10 13:45:42 -0700 | [diff] [blame] | 168 | |
| 169 | /** |
| 170 | * Indicates that an uninstall request are made and the actual result may come |
| 171 | * after some time. |
| 172 | */ |
| 173 | void deferCompleteDropAfterUninstallActivity(); |
| 174 | } |
| 175 | } |