blob: 184113413b15f7cdac73409a7329b6aa46c52fe7 [file] [log] [blame]
Sunny Goyalfa401a12015-04-10 13:45:42 -07001package com.android.launcher3;
2
Winson Chung1054d4e2018-03-05 19:39:21 +00003import static android.appwidget.AppWidgetManager.INVALID_APPWIDGET_ID;
4import static android.appwidget.AppWidgetProviderInfo.WIDGET_FEATURE_RECONFIGURABLE;
Samuel Fufacc1e1072019-09-06 16:19:11 -07005
Sunny Goyal076839c2017-10-30 13:52:20 -07006import static com.android.launcher3.ItemInfoWithIcon.FLAG_SYSTEM_MASK;
7import static com.android.launcher3.ItemInfoWithIcon.FLAG_SYSTEM_NO;
Winson Chung1054d4e2018-03-05 19:39:21 +00008import static com.android.launcher3.LauncherSettings.Favorites.CONTAINER_DESKTOP;
Samuel Fufa5cf3e862020-02-03 20:22:54 -08009import static com.android.launcher3.accessibility.LauncherAccessibilityDelegate.DISMISS_PREDICTION;
Winson Chung1054d4e2018-03-05 19:39:21 +000010import static com.android.launcher3.accessibility.LauncherAccessibilityDelegate.RECONFIGURE;
11import static com.android.launcher3.accessibility.LauncherAccessibilityDelegate.UNINSTALL;
Sunny Goyal076839c2017-10-30 13:52:20 -070012
Winson Chung1054d4e2018-03-05 19:39:21 +000013import android.appwidget.AppWidgetHostView;
Winson Chung1054d4e2018-03-05 19:39:21 +000014import android.appwidget.AppWidgetProviderInfo;
Sunny Goyalfa401a12015-04-10 13:45:42 -070015import android.content.ComponentName;
16import android.content.Context;
Sunny Goyald5bd67d2016-03-11 01:10:19 -080017import android.content.Intent;
Sunny Goyal8e0e1d72016-10-10 10:41:41 -070018import android.content.pm.ApplicationInfo;
Sunny Goyal3e9be432017-01-05 15:22:41 -080019import android.content.pm.LauncherActivityInfo;
Sunny Goyale7b00122019-10-02 16:13:34 -070020import android.content.pm.LauncherApps;
Sunny Goyalf2db2532017-03-01 17:27:16 -080021import android.content.pm.PackageManager;
Sunny Goyald5bd67d2016-03-11 01:10:19 -080022import android.net.Uri;
Sunny Goyalfa401a12015-04-10 13:45:42 -070023import android.os.Bundle;
Sunny Goyal7c74e4a2016-12-15 15:53:17 -080024import android.os.UserHandle;
Sunny Goyalfa401a12015-04-10 13:45:42 -070025import android.os.UserManager;
Sunny Goyal0236d0b2017-10-24 14:54:30 -070026import android.util.ArrayMap;
Sunny Goyalfa401a12015-04-10 13:45:42 -070027import android.util.AttributeSet;
Jon Mirandac56e3ff2017-08-23 12:13:24 -070028import android.util.Log;
Sunny Goyal3dce5f32017-10-05 11:40:05 -070029import android.view.View;
Sunny Goyald5bd67d2016-03-11 01:10:19 -080030import android.widget.Toast;
Sunny Goyalaa8ef112015-06-12 20:04:41 -070031
Sunny Goyal3dce5f32017-10-05 11:40:05 -070032import com.android.launcher3.Launcher.OnResumeCallback;
Samuel Fufa5cf3e862020-02-03 20:22:54 -080033import com.android.launcher3.config.FeatureFlags;
Sunny Goyal3dce5f32017-10-05 11:40:05 -070034import com.android.launcher3.dragndrop.DragOptions;
Samuel Fufacc1e1072019-09-06 16:19:11 -070035import com.android.launcher3.logging.FileLog;
Mehdi Alizadehbda47cf2018-05-01 19:26:05 -070036import com.android.launcher3.logging.LoggerUtils;
Samuel Fufa5cf3e862020-02-03 20:22:54 -080037import com.android.launcher3.model.AppLaunchTracker;
Mehdi Alizadehbda47cf2018-05-01 19:26:05 -070038import com.android.launcher3.userevent.nano.LauncherLogProto.ControlType;
Sunny Goyal3dce5f32017-10-05 11:40:05 -070039import com.android.launcher3.userevent.nano.LauncherLogProto.Target;
Sunny Goyale7b00122019-10-02 16:13:34 -070040import com.android.launcher3.util.PackageManagerHelper;
Winson Chung1054d4e2018-03-05 19:39:21 +000041import com.android.launcher3.util.Themes;
Sunny Goyalfa401a12015-04-10 13:45:42 -070042
Jon Mirandac56e3ff2017-08-23 12:13:24 -070043import java.net.URISyntaxException;
44
Winson Chung1054d4e2018-03-05 19:39:21 +000045/**
46 * Drop target which provides a secondary option for an item.
47 * For app targets: shows as uninstall
48 * For configurable widgets: shows as setup
Samuel Fufa5cf3e862020-02-03 20:22:54 -080049 * For predicted app icons: don't suggest app
Winson Chung1054d4e2018-03-05 19:39:21 +000050 */
51public class SecondaryDropTarget extends ButtonDropTarget implements OnAlarmListener {
Sunny Goyalfa401a12015-04-10 13:45:42 -070052
Winson Chung1054d4e2018-03-05 19:39:21 +000053 private static final String TAG = "SecondaryDropTarget";
Sunny Goyal0236d0b2017-10-24 14:54:30 -070054
55 private static final long CACHE_EXPIRE_TIMEOUT = 5000;
56 private final ArrayMap<UserHandle, Boolean> mUninstallDisabledCache = new ArrayMap<>(1);
57
58 private final Alarm mCacheExpireAlarm;
vadimt3d64ffd2020-02-27 11:28:47 -080059 private boolean mHadPendingAlarm;
Jon Mirandac56e3ff2017-08-23 12:13:24 -070060
Mehdi Alizadeh8517b692018-04-23 15:55:26 -070061 protected int mCurrentAccessibilityAction = -1;
Winson Chung1054d4e2018-03-05 19:39:21 +000062 public SecondaryDropTarget(Context context, AttributeSet attrs) {
Sunny Goyalfa401a12015-04-10 13:45:42 -070063 this(context, attrs, 0);
64 }
65
Winson Chung1054d4e2018-03-05 19:39:21 +000066 public SecondaryDropTarget(Context context, AttributeSet attrs, int defStyle) {
Sunny Goyalfa401a12015-04-10 13:45:42 -070067 super(context, attrs, defStyle);
Sunny Goyal0236d0b2017-10-24 14:54:30 -070068
69 mCacheExpireAlarm = new Alarm();
Sunny Goyalfa401a12015-04-10 13:45:42 -070070 }
71
72 @Override
vadimt3d64ffd2020-02-27 11:28:47 -080073 protected void onAttachedToWindow() {
74 super.onAttachedToWindow();
75 if (mHadPendingAlarm) {
76 mCacheExpireAlarm.setAlarm(CACHE_EXPIRE_TIMEOUT);
vadimt908e9bb2020-02-28 18:02:30 -080077 mCacheExpireAlarm.setOnAlarmListener(this);
vadimt3d64ffd2020-02-27 11:28:47 -080078 mHadPendingAlarm = false;
79 }
80 }
81
82 @Override
83 protected void onDetachedFromWindow() {
84 super.onDetachedFromWindow();
85 if (mCacheExpireAlarm.alarmPending()) {
86 mCacheExpireAlarm.cancelAlarm();
vadimt908e9bb2020-02-28 18:02:30 -080087 mCacheExpireAlarm.setOnAlarmListener(null);
vadimt3d64ffd2020-02-27 11:28:47 -080088 mHadPendingAlarm = true;
89 }
90 }
91
92 @Override
Sunny Goyalfa401a12015-04-10 13:45:42 -070093 protected void onFinishInflate() {
94 super.onFinishInflate();
Winson Chung1054d4e2018-03-05 19:39:21 +000095 setupUi(UNINSTALL);
Sunny Goyalda1dfa32017-04-26 22:34:49 -070096 }
97
Mehdi Alizadeh8517b692018-04-23 15:55:26 -070098 protected void setupUi(int action) {
Winson Chung1054d4e2018-03-05 19:39:21 +000099 if (action == mCurrentAccessibilityAction) {
100 return;
101 }
102 mCurrentAccessibilityAction = action;
103
104 if (action == UNINSTALL) {
105 mHoverColor = getResources().getColor(R.color.uninstall_target_hover_tint);
106 setDrawable(R.drawable.ic_uninstall_shadow);
107 updateText(R.string.uninstall_drop_target_label);
Samuel Fufa5cf3e862020-02-03 20:22:54 -0800108 } else if (action == DISMISS_PREDICTION) {
109 mHoverColor = Themes.getColorAccent(getContext());
110 setDrawable(R.drawable.ic_block);
111 updateText(R.string.dismiss_prediction_label);
112 } else if (action == RECONFIGURE) {
Winson Chung1054d4e2018-03-05 19:39:21 +0000113 mHoverColor = Themes.getColorAccent(getContext());
114 setDrawable(R.drawable.ic_setup_shadow);
115 updateText(R.string.gadget_setup_text);
116 }
Sunny Goyalfa401a12015-04-10 13:45:42 -0700117 }
118
119 @Override
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700120 public void onAlarm(Alarm alarm) {
121 mUninstallDisabledCache.clear();
Sunny Goyal1a70cef2015-04-22 11:29:51 -0700122 }
123
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700124 @Override
125 public int getAccessibilityAction() {
Winson Chung1054d4e2018-03-05 19:39:21 +0000126 return mCurrentAccessibilityAction;
127 }
128
129 @Override
Mehdi Alizadehbda47cf2018-05-01 19:26:05 -0700130 public Target getDropTargetForLogging() {
131 Target t = LoggerUtils.newTarget(Target.Type.CONTROL);
Samuel Fufa5cf3e862020-02-03 20:22:54 -0800132 if (mCurrentAccessibilityAction == UNINSTALL) {
133 t.controlType = ControlType.UNINSTALL_TARGET;
134 } else if (mCurrentAccessibilityAction == DISMISS_PREDICTION) {
135 t.controlType = ControlType.DISMISS_PREDICTION;
136 } else {
137 t.controlType = ControlType.SETTINGS_BUTTON;
138 }
Mehdi Alizadehbda47cf2018-05-01 19:26:05 -0700139 return t;
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700140 }
141
142 @Override
143 protected boolean supportsDrop(ItemInfo info) {
Winson Chung1054d4e2018-03-05 19:39:21 +0000144 return supportsAccessibilityDrop(info, getViewUnderDrag(info));
145 }
146
147 @Override
148 public boolean supportsAccessibilityDrop(ItemInfo info, View view) {
149 if (view instanceof AppWidgetHostView) {
150 if (getReconfigurableWidgetId(view) != INVALID_APPWIDGET_ID) {
151 setupUi(RECONFIGURE);
152 return true;
153 }
154 return false;
Samuel Fufa5cf3e862020-02-03 20:22:54 -0800155 } else if (FeatureFlags.ENABLE_PREDICTION_DISMISS.get() && info.isPredictedItem()) {
156 setupUi(DISMISS_PREDICTION);
157 return true;
Winson Chung1054d4e2018-03-05 19:39:21 +0000158 }
159
160 setupUi(UNINSTALL);
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700161 Boolean uninstallDisabled = mUninstallDisabledCache.get(info.user);
162 if (uninstallDisabled == null) {
163 UserManager userManager =
164 (UserManager) getContext().getSystemService(Context.USER_SERVICE);
165 Bundle restrictions = userManager.getUserRestrictions(info.user);
166 uninstallDisabled = restrictions.getBoolean(UserManager.DISALLOW_APPS_CONTROL, false)
Sunny Goyal6b0aa872017-09-29 07:54:37 -0700167 || restrictions.getBoolean(UserManager.DISALLOW_UNINSTALL_APPS, false);
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700168 mUninstallDisabledCache.put(info.user, uninstallDisabled);
Sunny Goyal6b0aa872017-09-29 07:54:37 -0700169 }
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700170 // Cancel any pending alarm and set cache expiry after some time
171 mCacheExpireAlarm.setAlarm(CACHE_EXPIRE_TIMEOUT);
vadimt908e9bb2020-02-28 18:02:30 -0800172 mCacheExpireAlarm.setOnAlarmListener(this);
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700173 if (uninstallDisabled) {
Sunny Goyala52ecb02016-12-16 15:04:51 -0800174 return false;
Sunny Goyalfa401a12015-04-10 13:45:42 -0700175 }
176
Sunny Goyal076839c2017-10-30 13:52:20 -0700177 if (info instanceof ItemInfoWithIcon) {
178 ItemInfoWithIcon iconInfo = (ItemInfoWithIcon) info;
179 if ((iconInfo.runtimeStatusFlags & FLAG_SYSTEM_MASK) != 0) {
180 return (iconInfo.runtimeStatusFlags & FLAG_SYSTEM_NO) != 0;
Sunny Goyal6b0aa872017-09-29 07:54:37 -0700181 }
182 }
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700183 return getUninstallTarget(info) != null;
Sunny Goyalfa401a12015-04-10 13:45:42 -0700184 }
185
186 /**
Sunny Goyal8e0e1d72016-10-10 10:41:41 -0700187 * @return the component name that should be uninstalled or null.
Sunny Goyalfa401a12015-04-10 13:45:42 -0700188 */
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700189 private ComponentName getUninstallTarget(ItemInfo item) {
Sunny Goyal8e0e1d72016-10-10 10:41:41 -0700190 Intent intent = null;
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800191 UserHandle user = null;
Sunny Goyal1fe0c2c2017-08-15 12:54:42 -0700192 if (item != null &&
Sunny Goyalc5939392018-12-07 11:43:47 -0800193 item.itemType == LauncherSettings.Favorites.ITEM_TYPE_APPLICATION) {
Sunny Goyal1fe0c2c2017-08-15 12:54:42 -0700194 intent = item.getIntent();
195 user = item.user;
Sunny Goyal8e0e1d72016-10-10 10:41:41 -0700196 }
197 if (intent != null) {
Sunny Goyale7b00122019-10-02 16:13:34 -0700198 LauncherActivityInfo info = mLauncher.getSystemService(LauncherApps.class)
Sunny Goyal8e0e1d72016-10-10 10:41:41 -0700199 .resolveActivity(intent, user);
200 if (info != null
201 && (info.getApplicationInfo().flags & ApplicationInfo.FLAG_SYSTEM) == 0) {
202 return info.getComponentName();
Sunny Goyalfa401a12015-04-10 13:45:42 -0700203 }
204 }
205 return null;
206 }
207
208 @Override
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700209 public void onDrop(DragObject d, DragOptions options) {
210 // Defer onComplete
Sunny Goyal1797af42017-10-06 13:29:57 -0700211 d.dragSource = new DeferredOnComplete(d.dragSource, getContext());
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700212 super.onDrop(d, options);
Sunny Goyalfa401a12015-04-10 13:45:42 -0700213 }
214
215 @Override
Sunny Goyal0f76b562016-12-13 19:37:10 -0800216 public void completeDrop(final DragObject d) {
Winson Chung1054d4e2018-03-05 19:39:21 +0000217 ComponentName target = performDropAction(getViewUnderDrag(d.dragInfo), d.dragInfo);
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700218 if (d.dragSource instanceof DeferredOnComplete) {
219 DeferredOnComplete deferred = (DeferredOnComplete) d.dragSource;
220 if (target != null) {
221 deferred.mPackageName = target.getPackageName();
Winson Chung0b70cd42019-06-17 22:34:33 -0700222 mLauncher.addOnResumeCallback(deferred);
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700223 } else {
224 deferred.sendFailure();
225 }
226 }
Tony Wickham734dfbe2015-09-16 16:22:36 -0700227 }
228
Winson Chung1054d4e2018-03-05 19:39:21 +0000229 private View getViewUnderDrag(ItemInfo info) {
230 if (info instanceof LauncherAppWidgetInfo && info.container == CONTAINER_DESKTOP &&
231 mLauncher.getWorkspace().getDragInfo() != null) {
232 return mLauncher.getWorkspace().getDragInfo().cell;
233 }
234 return null;
235 }
236
237 /**
238 * Verifies that the view is an reconfigurable widget and returns the corresponding widget Id,
239 * otherwise return {@code INVALID_APPWIDGET_ID}
240 */
241 private int getReconfigurableWidgetId(View view) {
242 if (!(view instanceof AppWidgetHostView)) {
243 return INVALID_APPWIDGET_ID;
244 }
245 AppWidgetHostView hostView = (AppWidgetHostView) view;
246 AppWidgetProviderInfo widgetInfo = hostView.getAppWidgetInfo();
247 if (widgetInfo == null || widgetInfo.configure == null) {
248 return INVALID_APPWIDGET_ID;
249 }
250 if ( (LauncherAppWidgetProviderInfo.fromProviderInfo(getContext(), widgetInfo)
251 .getWidgetFeatures() & WIDGET_FEATURE_RECONFIGURABLE) == 0) {
252 return INVALID_APPWIDGET_ID;
253 }
254 return hostView.getAppWidgetId();
255 }
256
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700257 /**
258 * Performs the drop action and returns the target component for the dragObject or null if
259 * the action was not performed.
260 */
Winson Chung1054d4e2018-03-05 19:39:21 +0000261 protected ComponentName performDropAction(View view, ItemInfo info) {
262 if (mCurrentAccessibilityAction == RECONFIGURE) {
263 int widgetId = getReconfigurableWidgetId(view);
264 if (widgetId != INVALID_APPWIDGET_ID) {
265 mLauncher.getAppWidgetHost().startConfigActivity(mLauncher, widgetId, -1);
266 }
267 return null;
268 }
Samuel Fufa5cf3e862020-02-03 20:22:54 -0800269 if (mCurrentAccessibilityAction == DISMISS_PREDICTION) {
270 AppLaunchTracker.INSTANCE.get(getContext()).onDismissApp(info.getTargetComponent(),
271 info.user, AppLaunchTracker.CONTAINER_PREDICTIONS);
272 return null;
273 }
Winson Chung1054d4e2018-03-05 19:39:21 +0000274 // else: mCurrentAccessibilityAction == UNINSTALL
275
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700276 ComponentName cn = getUninstallTarget(info);
Sunny Goyal8e0e1d72016-10-10 10:41:41 -0700277 if (cn == null) {
Sunny Goyald5bd67d2016-03-11 01:10:19 -0800278 // System applications cannot be installed. For now, show a toast explaining that.
279 // We may give them the option of disabling apps this way.
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700280 Toast.makeText(mLauncher, R.string.uninstall_system_app_text, Toast.LENGTH_SHORT).show();
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700281 return null;
282 }
283 try {
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700284 Intent i = Intent.parseUri(mLauncher.getString(R.string.delete_package_intent), 0)
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700285 .setData(Uri.fromParts("package", cn.getPackageName(), cn.getClassName()))
286 .putExtra(Intent.EXTRA_USER, info.user);
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700287 mLauncher.startActivity(i);
Samuel Fufacc1e1072019-09-06 16:19:11 -0700288 FileLog.d(TAG, "start uninstall activity " + cn.getPackageName());
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700289 return cn;
290 } catch (URISyntaxException e) {
291 Log.e(TAG, "Failed to parse intent to start uninstall activity for item=" + info);
292 return null;
293 }
294 }
295
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700296 @Override
297 public void onAccessibilityDrop(View view, ItemInfo item) {
Winson Chung1054d4e2018-03-05 19:39:21 +0000298 performDropAction(view, item);
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700299 }
300
301 /**
302 * A wrapper around {@link DragSource} which delays the {@link #onDropCompleted} action until
303 * {@link #onLauncherResume}
304 */
305 private class DeferredOnComplete implements DragSource, OnResumeCallback {
306
307 private final DragSource mOriginal;
308 private final Context mContext;
309
310 private String mPackageName;
311 private DragObject mDragObject;
312
313 public DeferredOnComplete(DragSource original, Context context) {
314 mOriginal = original;
315 mContext = context;
316 }
317
318 @Override
Sunny Goyal1797af42017-10-06 13:29:57 -0700319 public void onDropCompleted(View target, DragObject d,
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700320 boolean success) {
321 mDragObject = d;
322 }
323
324 @Override
325 public void fillInLogContainerData(View v, ItemInfo info, Target target,
326 Target targetParent) {
327 mOriginal.fillInLogContainerData(v, info, target, targetParent);
328 }
329
330 @Override
331 public void onLauncherResume() {
332 // We use MATCH_UNINSTALLED_PACKAGES as the app can be on SD card as well.
Sunny Goyale7b00122019-10-02 16:13:34 -0700333 if (new PackageManagerHelper(mContext).getApplicationInfo(mPackageName,
334 mDragObject.dragInfo.user, PackageManager.MATCH_UNINSTALLED_PACKAGES) == null) {
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700335 mDragObject.dragSource = mOriginal;
Winson Chung1054d4e2018-03-05 19:39:21 +0000336 mOriginal.onDropCompleted(SecondaryDropTarget.this, mDragObject, true);
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700337 } else {
338 sendFailure();
Jon Mirandac56e3ff2017-08-23 12:13:24 -0700339 }
Sunny Goyalfa401a12015-04-10 13:45:42 -0700340 }
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700341
342 public void sendFailure() {
343 mDragObject.dragSource = mOriginal;
344 mDragObject.cancelled = true;
Winson Chung1054d4e2018-03-05 19:39:21 +0000345 mOriginal.onDropCompleted(SecondaryDropTarget.this, mDragObject, false);
Sunny Goyald5bd67d2016-03-11 01:10:19 -0800346 }
Sunny Goyalfa401a12015-04-10 13:45:42 -0700347 }
348}