blob: 1dbe195da38cf346924c4a55cff4afabc4656d3b [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;
Jon Mirandac56e3ff2017-08-23 12:13:24 -070059
Mehdi Alizadeh8517b692018-04-23 15:55:26 -070060 protected int mCurrentAccessibilityAction = -1;
Winson Chung1054d4e2018-03-05 19:39:21 +000061 public SecondaryDropTarget(Context context, AttributeSet attrs) {
Sunny Goyalfa401a12015-04-10 13:45:42 -070062 this(context, attrs, 0);
63 }
64
Winson Chung1054d4e2018-03-05 19:39:21 +000065 public SecondaryDropTarget(Context context, AttributeSet attrs, int defStyle) {
Sunny Goyalfa401a12015-04-10 13:45:42 -070066 super(context, attrs, defStyle);
Sunny Goyal0236d0b2017-10-24 14:54:30 -070067
68 mCacheExpireAlarm = new Alarm();
69 mCacheExpireAlarm.setOnAlarmListener(this);
Sunny Goyalfa401a12015-04-10 13:45:42 -070070 }
71
72 @Override
73 protected void onFinishInflate() {
74 super.onFinishInflate();
Winson Chung1054d4e2018-03-05 19:39:21 +000075 setupUi(UNINSTALL);
Sunny Goyalda1dfa32017-04-26 22:34:49 -070076 }
77
Mehdi Alizadeh8517b692018-04-23 15:55:26 -070078 protected void setupUi(int action) {
Winson Chung1054d4e2018-03-05 19:39:21 +000079 if (action == mCurrentAccessibilityAction) {
80 return;
81 }
82 mCurrentAccessibilityAction = action;
83
84 if (action == UNINSTALL) {
85 mHoverColor = getResources().getColor(R.color.uninstall_target_hover_tint);
86 setDrawable(R.drawable.ic_uninstall_shadow);
87 updateText(R.string.uninstall_drop_target_label);
Samuel Fufa5cf3e862020-02-03 20:22:54 -080088 } else if (action == DISMISS_PREDICTION) {
89 mHoverColor = Themes.getColorAccent(getContext());
90 setDrawable(R.drawable.ic_block);
91 updateText(R.string.dismiss_prediction_label);
92 } else if (action == RECONFIGURE) {
Winson Chung1054d4e2018-03-05 19:39:21 +000093 mHoverColor = Themes.getColorAccent(getContext());
94 setDrawable(R.drawable.ic_setup_shadow);
95 updateText(R.string.gadget_setup_text);
96 }
Sunny Goyalfa401a12015-04-10 13:45:42 -070097 }
98
99 @Override
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700100 public void onAlarm(Alarm alarm) {
101 mUninstallDisabledCache.clear();
Sunny Goyal1a70cef2015-04-22 11:29:51 -0700102 }
103
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700104 @Override
105 public int getAccessibilityAction() {
Winson Chung1054d4e2018-03-05 19:39:21 +0000106 return mCurrentAccessibilityAction;
107 }
108
109 @Override
Mehdi Alizadehbda47cf2018-05-01 19:26:05 -0700110 public Target getDropTargetForLogging() {
111 Target t = LoggerUtils.newTarget(Target.Type.CONTROL);
Samuel Fufa5cf3e862020-02-03 20:22:54 -0800112 if (mCurrentAccessibilityAction == UNINSTALL) {
113 t.controlType = ControlType.UNINSTALL_TARGET;
114 } else if (mCurrentAccessibilityAction == DISMISS_PREDICTION) {
115 t.controlType = ControlType.DISMISS_PREDICTION;
116 } else {
117 t.controlType = ControlType.SETTINGS_BUTTON;
118 }
Mehdi Alizadehbda47cf2018-05-01 19:26:05 -0700119 return t;
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700120 }
121
122 @Override
123 protected boolean supportsDrop(ItemInfo info) {
Winson Chung1054d4e2018-03-05 19:39:21 +0000124 return supportsAccessibilityDrop(info, getViewUnderDrag(info));
125 }
126
127 @Override
128 public boolean supportsAccessibilityDrop(ItemInfo info, View view) {
129 if (view instanceof AppWidgetHostView) {
130 if (getReconfigurableWidgetId(view) != INVALID_APPWIDGET_ID) {
131 setupUi(RECONFIGURE);
132 return true;
133 }
134 return false;
Samuel Fufa5cf3e862020-02-03 20:22:54 -0800135 } else if (FeatureFlags.ENABLE_PREDICTION_DISMISS.get() && info.isPredictedItem()) {
136 setupUi(DISMISS_PREDICTION);
137 return true;
Winson Chung1054d4e2018-03-05 19:39:21 +0000138 }
139
140 setupUi(UNINSTALL);
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700141 Boolean uninstallDisabled = mUninstallDisabledCache.get(info.user);
142 if (uninstallDisabled == null) {
143 UserManager userManager =
144 (UserManager) getContext().getSystemService(Context.USER_SERVICE);
145 Bundle restrictions = userManager.getUserRestrictions(info.user);
146 uninstallDisabled = restrictions.getBoolean(UserManager.DISALLOW_APPS_CONTROL, false)
Sunny Goyal6b0aa872017-09-29 07:54:37 -0700147 || restrictions.getBoolean(UserManager.DISALLOW_UNINSTALL_APPS, false);
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700148 mUninstallDisabledCache.put(info.user, uninstallDisabled);
Sunny Goyal6b0aa872017-09-29 07:54:37 -0700149 }
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700150 // Cancel any pending alarm and set cache expiry after some time
151 mCacheExpireAlarm.setAlarm(CACHE_EXPIRE_TIMEOUT);
152 if (uninstallDisabled) {
Sunny Goyala52ecb02016-12-16 15:04:51 -0800153 return false;
Sunny Goyalfa401a12015-04-10 13:45:42 -0700154 }
155
Sunny Goyal076839c2017-10-30 13:52:20 -0700156 if (info instanceof ItemInfoWithIcon) {
157 ItemInfoWithIcon iconInfo = (ItemInfoWithIcon) info;
158 if ((iconInfo.runtimeStatusFlags & FLAG_SYSTEM_MASK) != 0) {
159 return (iconInfo.runtimeStatusFlags & FLAG_SYSTEM_NO) != 0;
Sunny Goyal6b0aa872017-09-29 07:54:37 -0700160 }
161 }
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700162 return getUninstallTarget(info) != null;
Sunny Goyalfa401a12015-04-10 13:45:42 -0700163 }
164
165 /**
Sunny Goyal8e0e1d72016-10-10 10:41:41 -0700166 * @return the component name that should be uninstalled or null.
Sunny Goyalfa401a12015-04-10 13:45:42 -0700167 */
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700168 private ComponentName getUninstallTarget(ItemInfo item) {
Sunny Goyal8e0e1d72016-10-10 10:41:41 -0700169 Intent intent = null;
Sunny Goyal7c74e4a2016-12-15 15:53:17 -0800170 UserHandle user = null;
Sunny Goyal1fe0c2c2017-08-15 12:54:42 -0700171 if (item != null &&
Sunny Goyalc5939392018-12-07 11:43:47 -0800172 item.itemType == LauncherSettings.Favorites.ITEM_TYPE_APPLICATION) {
Sunny Goyal1fe0c2c2017-08-15 12:54:42 -0700173 intent = item.getIntent();
174 user = item.user;
Sunny Goyal8e0e1d72016-10-10 10:41:41 -0700175 }
176 if (intent != null) {
Sunny Goyale7b00122019-10-02 16:13:34 -0700177 LauncherActivityInfo info = mLauncher.getSystemService(LauncherApps.class)
Sunny Goyal8e0e1d72016-10-10 10:41:41 -0700178 .resolveActivity(intent, user);
179 if (info != null
180 && (info.getApplicationInfo().flags & ApplicationInfo.FLAG_SYSTEM) == 0) {
181 return info.getComponentName();
Sunny Goyalfa401a12015-04-10 13:45:42 -0700182 }
183 }
184 return null;
185 }
186
187 @Override
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700188 public void onDrop(DragObject d, DragOptions options) {
189 // Defer onComplete
Sunny Goyal1797af42017-10-06 13:29:57 -0700190 d.dragSource = new DeferredOnComplete(d.dragSource, getContext());
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700191 super.onDrop(d, options);
Sunny Goyalfa401a12015-04-10 13:45:42 -0700192 }
193
194 @Override
Sunny Goyal0f76b562016-12-13 19:37:10 -0800195 public void completeDrop(final DragObject d) {
Winson Chung1054d4e2018-03-05 19:39:21 +0000196 ComponentName target = performDropAction(getViewUnderDrag(d.dragInfo), d.dragInfo);
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700197 if (d.dragSource instanceof DeferredOnComplete) {
198 DeferredOnComplete deferred = (DeferredOnComplete) d.dragSource;
199 if (target != null) {
200 deferred.mPackageName = target.getPackageName();
Winson Chung0b70cd42019-06-17 22:34:33 -0700201 mLauncher.addOnResumeCallback(deferred);
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700202 } else {
203 deferred.sendFailure();
204 }
205 }
Tony Wickham734dfbe2015-09-16 16:22:36 -0700206 }
207
Winson Chung1054d4e2018-03-05 19:39:21 +0000208 private View getViewUnderDrag(ItemInfo info) {
209 if (info instanceof LauncherAppWidgetInfo && info.container == CONTAINER_DESKTOP &&
210 mLauncher.getWorkspace().getDragInfo() != null) {
211 return mLauncher.getWorkspace().getDragInfo().cell;
212 }
213 return null;
214 }
215
216 /**
217 * Verifies that the view is an reconfigurable widget and returns the corresponding widget Id,
218 * otherwise return {@code INVALID_APPWIDGET_ID}
219 */
220 private int getReconfigurableWidgetId(View view) {
221 if (!(view instanceof AppWidgetHostView)) {
222 return INVALID_APPWIDGET_ID;
223 }
224 AppWidgetHostView hostView = (AppWidgetHostView) view;
225 AppWidgetProviderInfo widgetInfo = hostView.getAppWidgetInfo();
226 if (widgetInfo == null || widgetInfo.configure == null) {
227 return INVALID_APPWIDGET_ID;
228 }
229 if ( (LauncherAppWidgetProviderInfo.fromProviderInfo(getContext(), widgetInfo)
230 .getWidgetFeatures() & WIDGET_FEATURE_RECONFIGURABLE) == 0) {
231 return INVALID_APPWIDGET_ID;
232 }
233 return hostView.getAppWidgetId();
234 }
235
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700236 /**
237 * Performs the drop action and returns the target component for the dragObject or null if
238 * the action was not performed.
239 */
Winson Chung1054d4e2018-03-05 19:39:21 +0000240 protected ComponentName performDropAction(View view, ItemInfo info) {
241 if (mCurrentAccessibilityAction == RECONFIGURE) {
242 int widgetId = getReconfigurableWidgetId(view);
243 if (widgetId != INVALID_APPWIDGET_ID) {
244 mLauncher.getAppWidgetHost().startConfigActivity(mLauncher, widgetId, -1);
245 }
246 return null;
247 }
Samuel Fufa5cf3e862020-02-03 20:22:54 -0800248 if (mCurrentAccessibilityAction == DISMISS_PREDICTION) {
249 AppLaunchTracker.INSTANCE.get(getContext()).onDismissApp(info.getTargetComponent(),
250 info.user, AppLaunchTracker.CONTAINER_PREDICTIONS);
251 return null;
252 }
Winson Chung1054d4e2018-03-05 19:39:21 +0000253 // else: mCurrentAccessibilityAction == UNINSTALL
254
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700255 ComponentName cn = getUninstallTarget(info);
Sunny Goyal8e0e1d72016-10-10 10:41:41 -0700256 if (cn == null) {
Sunny Goyald5bd67d2016-03-11 01:10:19 -0800257 // System applications cannot be installed. For now, show a toast explaining that.
258 // We may give them the option of disabling apps this way.
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700259 Toast.makeText(mLauncher, R.string.uninstall_system_app_text, Toast.LENGTH_SHORT).show();
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700260 return null;
261 }
262 try {
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700263 Intent i = Intent.parseUri(mLauncher.getString(R.string.delete_package_intent), 0)
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700264 .setData(Uri.fromParts("package", cn.getPackageName(), cn.getClassName()))
265 .putExtra(Intent.EXTRA_USER, info.user);
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700266 mLauncher.startActivity(i);
Samuel Fufacc1e1072019-09-06 16:19:11 -0700267 FileLog.d(TAG, "start uninstall activity " + cn.getPackageName());
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700268 return cn;
269 } catch (URISyntaxException e) {
270 Log.e(TAG, "Failed to parse intent to start uninstall activity for item=" + info);
271 return null;
272 }
273 }
274
Sunny Goyal0236d0b2017-10-24 14:54:30 -0700275 @Override
276 public void onAccessibilityDrop(View view, ItemInfo item) {
Winson Chung1054d4e2018-03-05 19:39:21 +0000277 performDropAction(view, item);
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700278 }
279
280 /**
281 * A wrapper around {@link DragSource} which delays the {@link #onDropCompleted} action until
282 * {@link #onLauncherResume}
283 */
284 private class DeferredOnComplete implements DragSource, OnResumeCallback {
285
286 private final DragSource mOriginal;
287 private final Context mContext;
288
289 private String mPackageName;
290 private DragObject mDragObject;
291
292 public DeferredOnComplete(DragSource original, Context context) {
293 mOriginal = original;
294 mContext = context;
295 }
296
297 @Override
Sunny Goyal1797af42017-10-06 13:29:57 -0700298 public void onDropCompleted(View target, DragObject d,
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700299 boolean success) {
300 mDragObject = d;
301 }
302
303 @Override
304 public void fillInLogContainerData(View v, ItemInfo info, Target target,
305 Target targetParent) {
306 mOriginal.fillInLogContainerData(v, info, target, targetParent);
307 }
308
309 @Override
310 public void onLauncherResume() {
311 // We use MATCH_UNINSTALLED_PACKAGES as the app can be on SD card as well.
Sunny Goyale7b00122019-10-02 16:13:34 -0700312 if (new PackageManagerHelper(mContext).getApplicationInfo(mPackageName,
313 mDragObject.dragInfo.user, PackageManager.MATCH_UNINSTALLED_PACKAGES) == null) {
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700314 mDragObject.dragSource = mOriginal;
Winson Chung1054d4e2018-03-05 19:39:21 +0000315 mOriginal.onDropCompleted(SecondaryDropTarget.this, mDragObject, true);
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700316 } else {
317 sendFailure();
Jon Mirandac56e3ff2017-08-23 12:13:24 -0700318 }
Sunny Goyalfa401a12015-04-10 13:45:42 -0700319 }
Sunny Goyal3dce5f32017-10-05 11:40:05 -0700320
321 public void sendFailure() {
322 mDragObject.dragSource = mOriginal;
323 mDragObject.cancelled = true;
Winson Chung1054d4e2018-03-05 19:39:21 +0000324 mOriginal.onDropCompleted(SecondaryDropTarget.this, mDragObject, false);
Sunny Goyald5bd67d2016-03-11 01:10:19 -0800325 }
Sunny Goyalfa401a12015-04-10 13:45:42 -0700326 }
327}