blob: 7d15f7b7c00d8bc06c55725cd5652d174d8e0e28 [file] [log] [blame]
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001/*
2 * Copyright (C) 2008 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Daniel Sandler325dc232013-06-05 22:57:57 -040017package com.android.launcher3;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080018
Sunny Goyaleaf7a952020-07-29 16:54:20 -070019import static android.animation.ValueAnimator.areAnimatorsEnabled;
20
Kateryna Ivanova71203732023-05-24 15:09:00 +000021import static com.android.app.animation.Interpolators.DECELERATE_1_5;
Charlie Anderson438d4052023-04-25 14:38:57 -040022import static com.android.launcher3.LauncherState.EDIT_MODE;
Tony Wickham0ac045f2021-11-03 13:17:02 -070023import static com.android.launcher3.dragndrop.DraggableView.DRAGGABLE_ICON;
Tony Wickham12784902021-11-03 14:02:10 -070024import static com.android.launcher3.icons.IconNormalizer.ICON_VISIBLE_AREA_FACTOR;
Sunny Goyal82dfc152023-02-24 16:50:09 -080025import static com.android.launcher3.util.MultiTranslateDelegate.INDEX_REORDER_BOUNCE_OFFSET;
26import static com.android.launcher3.util.MultiTranslateDelegate.INDEX_REORDER_PREVIEW_OFFSET;
Sunny Goyalf0b6db72018-08-13 16:10:14 -070027
Joe Onorato4be866d2010-10-10 11:26:02 -070028import android.animation.Animator;
Michael Jurka629758f2012-06-14 16:18:21 -070029import android.animation.AnimatorListenerAdapter;
Sunny Goyal849c6a22018-08-08 16:33:46 -070030import android.animation.ObjectAnimator;
Chet Haase00397b12010-10-07 11:13:10 -070031import android.animation.TimeInterpolator;
Patrick Dubroyde7658b2010-09-27 11:15:43 -070032import android.animation.ValueAnimator;
33import android.animation.ValueAnimator.AnimatorUpdateListener;
Sunny Goyal726bee72018-03-05 12:54:24 -080034import android.annotation.SuppressLint;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080035import android.content.Context;
Joe Onorato79e56262009-09-21 15:23:04 -040036import android.content.res.Resources;
Sunny Goyalc13403c2016-11-18 23:44:48 -080037import android.content.res.TypedArray;
Winson Chungaafa03c2010-06-11 17:34:16 -070038import android.graphics.Canvas;
Andrew Flynn0dca1ec2012-02-29 13:33:22 -080039import android.graphics.Color;
Joe Onorato4be866d2010-10-10 11:26:02 -070040import android.graphics.Paint;
Patrick Dubroyde7658b2010-09-27 11:15:43 -070041import android.graphics.Point;
Adam Cohend9162062020-03-24 16:35:35 -070042import android.graphics.PointF;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080043import android.graphics.Rect;
Adam Cohenf7ca3b42021-02-22 11:03:58 -080044import android.graphics.RectF;
Patrick Dubroy6569f2c2010-07-12 14:25:18 -070045import android.graphics.drawable.Drawable;
Adam Cohen1462de32012-07-24 22:34:36 -070046import android.os.Parcelable;
Rajeev Kumar9962dbe2017-06-12 12:16:20 -070047import android.util.ArrayMap;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080048import android.util.AttributeSet;
Adam Cohenf7ca3b42021-02-22 11:03:58 -080049import android.util.FloatProperty;
Joe Onorato4be866d2010-10-10 11:26:02 -070050import android.util.Log;
Sunny Goyal849c6a22018-08-08 16:33:46 -070051import android.util.Property;
Adam Cohen1462de32012-07-24 22:34:36 -070052import android.util.SparseArray;
The Android Open Source Project31dd5032009-03-03 19:32:27 -080053import android.view.MotionEvent;
54import android.view.View;
55import android.view.ViewDebug;
56import android.view.ViewGroup;
Adam Cohenc9735cf2015-01-23 16:11:55 -080057import android.view.accessibility.AccessibilityEvent;
Sunny Goyal5bc6b6f2017-10-26 15:36:10 -070058
vadimt04f356f2019-02-14 18:46:36 -080059import androidx.annotation.IntDef;
Sebastian Franco5c8f8682023-11-14 09:52:41 -060060import androidx.annotation.Nullable;
Sebastian Franco9ea36d42023-09-21 13:56:42 -070061import androidx.annotation.Px;
Adam Cohenf7ca3b42021-02-22 11:03:58 -080062import androidx.core.graphics.ColorUtils;
vadimt04f356f2019-02-14 18:46:36 -080063import androidx.core.view.ViewCompat;
64
Kateryna Ivanova71203732023-05-24 15:09:00 +000065import com.android.app.animation.Interpolators;
Sunny Goyalaa8ef112015-06-12 20:04:41 -070066import com.android.launcher3.LauncherSettings.Favorites;
Sunny Goyale9b651e2015-04-24 11:44:51 -070067import com.android.launcher3.accessibility.DragAndDropAccessibilityDelegate;
Sebastian Francod4682992022-10-05 13:03:09 -050068import com.android.launcher3.celllayout.CellLayoutLayoutParams;
Sunny Goyal669b71f2023-01-27 14:37:07 -080069import com.android.launcher3.celllayout.CellPosMapper.CellPos;
Sebastian Franco5f0af4f2023-11-21 10:45:45 -060070import com.android.launcher3.celllayout.DelegatedCellDrawing;
71import com.android.launcher3.celllayout.ItemConfiguration;
Sebastian Francoe4c03452022-12-27 14:50:02 -060072import com.android.launcher3.celllayout.ReorderAlgorithm;
Sunny Goyal3d706ad2017-03-06 16:56:39 -080073import com.android.launcher3.config.FeatureFlags;
Tony Wickham0ac045f2021-11-03 13:17:02 -070074import com.android.launcher3.dragndrop.DraggableView;
Jon Mirandaa0233f72017-06-22 18:34:45 -070075import com.android.launcher3.folder.PreviewBackground;
Sunny Goyale396abf2020-04-06 15:11:17 -070076import com.android.launcher3.model.data.ItemInfo;
Sebastian Francof153d912022-04-22 16:15:27 -050077import com.android.launcher3.model.data.LauncherAppWidgetInfo;
Sunny Goyalff4ba2d2016-04-02 14:12:34 -070078import com.android.launcher3.util.CellAndSpan;
79import com.android.launcher3.util.GridOccupancy;
Sunny Goyal82dfc152023-02-24 16:50:09 -080080import com.android.launcher3.util.MultiTranslateDelegate;
Sunny Goyale2fd14b2015-08-27 17:45:46 -070081import com.android.launcher3.util.ParcelableSparseArray;
Sunny Goyal9b29ca52017-02-17 10:39:44 -080082import com.android.launcher3.util.Themes;
Adam Cohen091440a2015-03-18 14:16:05 -070083import com.android.launcher3.util.Thunk;
Sunny Goyalab770a12018-11-14 15:17:26 -080084import com.android.launcher3.views.ActivityContext;
Steven Ng32427202021-04-19 18:12:12 +010085import com.android.launcher3.widget.LauncherAppWidgetHostView;
Sunny Goyal5bc6b6f2017-10-26 15:36:10 -070086
Sunny Goyalc13403c2016-11-18 23:44:48 -080087import java.lang.annotation.Retention;
88import java.lang.annotation.RetentionPolicy;
Adam Cohen69ce2e52011-07-03 19:25:21 -070089import java.util.ArrayList;
Adam Cohenc0dcf592011-06-01 15:30:43 -070090import java.util.Arrays;
Adam Cohend41fbf52012-02-16 23:53:59 -080091import java.util.Stack;
Adam Cohenc0dcf592011-06-01 15:30:43 -070092
Sunny Goyalc4d32012020-04-03 17:10:11 -070093public class CellLayout extends ViewGroup {
Tony Wickhama0628cc2015-10-14 15:23:04 -070094 private static final String TAG = "CellLayout";
95 private static final boolean LOGD = false;
Winson Chungaafa03c2010-06-11 17:34:16 -070096
Tony Wickhamec6fd6f2023-03-11 02:08:57 +000097 /** The color of the "leave-behind" shape when a folder is opened from Hotseat. */
98 private static final int FOLDER_LEAVE_BEHIND_COLOR = Color.argb(160, 245, 245, 245);
99
Sunny Goyalab770a12018-11-14 15:17:26 -0800100 protected final ActivityContext mActivity;
Sunny Goyal4ffec482016-02-09 11:28:52 -0800101 @ViewDebug.ExportedProperty(category = "launcher")
Adam Cohen091440a2015-03-18 14:16:05 -0700102 @Thunk int mCellWidth;
Sunny Goyal4ffec482016-02-09 11:28:52 -0800103 @ViewDebug.ExportedProperty(category = "launcher")
Adam Cohen091440a2015-03-18 14:16:05 -0700104 @Thunk int mCellHeight;
Winson Chung11a1a532013-09-13 11:14:45 -0700105 private int mFixedCellWidth;
106 private int mFixedCellHeight;
Jon Miranda228877d2021-02-09 11:05:00 -0500107 @ViewDebug.ExportedProperty(category = "launcher")
Sebastian Franco25423862023-03-10 10:50:37 -0800108 protected Point mBorderSpace;
Winson Chungaafa03c2010-06-11 17:34:16 -0700109
Sunny Goyal4ffec482016-02-09 11:28:52 -0800110 @ViewDebug.ExportedProperty(category = "launcher")
Sebastian Franco09589322022-11-02 15:25:58 -0700111 protected int mCountX;
Sunny Goyal4ffec482016-02-09 11:28:52 -0800112 @ViewDebug.ExportedProperty(category = "launcher")
Sebastian Franco09589322022-11-02 15:25:58 -0700113 protected int mCountY;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800114
Adam Cohen917e3882013-10-31 15:03:35 -0700115 private boolean mDropPending = false;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800116
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700117 // These are temporary variables to prevent having to allocate a new object just to
118 // return an (x, y) value from helper functions. Do NOT use them to maintain other state.
Adam Cohen091440a2015-03-18 14:16:05 -0700119 @Thunk final int[] mTmpPoint = new int[2];
Sunny Goyal2805e632015-05-20 15:35:32 -0700120 @Thunk final int[] mTempLocation = new int[2];
Sebastian Franco0dd5db82023-10-13 11:09:21 -0700121
122 @Thunk final Rect mTempOnDrawCellToRect = new Rect();
Adam Cohend9162062020-03-24 16:35:35 -0700123 final PointF mTmpPointF = new PointF();
Patrick Dubroy6569f2c2010-07-12 14:25:18 -0700124
Sebastian Franco09589322022-11-02 15:25:58 -0700125 protected GridOccupancy mOccupied;
Sebastian Francoe4c03452022-12-27 14:50:02 -0600126 public GridOccupancy mTmpOccupied;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800127
Michael Jurkadee05892010-07-27 10:01:56 -0700128 private OnTouchListener mInterceptTouchListener;
129
Samuel Fufa1e2d0042019-11-18 17:12:46 -0800130 private final ArrayList<DelegatedCellDrawing> mDelegatedCellDrawings = new ArrayList<>();
Jon Mirandaa0233f72017-06-22 18:34:45 -0700131 final PreviewBackground mFolderLeaveBehind = new PreviewBackground();
Adam Cohen69ce2e52011-07-03 19:25:21 -0700132
Sunny Goyalf5440cb2016-12-14 15:13:00 -0800133 private static final int[] BACKGROUND_STATE_ACTIVE = new int[] { android.R.attr.state_active };
Sunny Goyale15e2a82017-12-15 13:05:42 -0800134 private static final int[] BACKGROUND_STATE_DEFAULT = EMPTY_STATE_SET;
Sebastian Franco09589322022-11-02 15:25:58 -0700135 protected final Drawable mBackground;
Sunny Goyal2805e632015-05-20 15:35:32 -0700136
Adam Cohenf0f4eda2013-06-06 21:27:03 -0700137 // These values allow a fixed measurement to be set on the CellLayout.
138 private int mFixedWidth = -1;
139 private int mFixedHeight = -1;
140
Michael Jurka33945b22010-12-21 18:19:38 -0800141 // If we're actively dragging something over this screen, mIsDragOverlapping is true
142 private boolean mIsDragOverlapping = false;
Patrick Dubroy6569f2c2010-07-12 14:25:18 -0700143
Winson Chung150fbab2010-09-29 17:14:26 -0700144 // These arrays are used to implement the drag visualization on x-large screens.
Joe Onorato4be866d2010-10-10 11:26:02 -0700145 // They are used as circular arrays, indexed by mDragOutlineCurrent.
Sebastian Francod4682992022-10-05 13:03:09 -0500146 @Thunk final CellLayoutLayoutParams[] mDragOutlines = new CellLayoutLayoutParams[4];
Rajeev Kumar9962dbe2017-06-12 12:16:20 -0700147 @Thunk final float[] mDragOutlineAlphas = new float[mDragOutlines.length];
148 private final InterruptibleInOutAnimator[] mDragOutlineAnims =
Joe Onorato4be866d2010-10-10 11:26:02 -0700149 new InterruptibleInOutAnimator[mDragOutlines.length];
Winson Chung150fbab2010-09-29 17:14:26 -0700150
151 // Used as an index into the above 3 arrays; indicates which is the most current value.
Joe Onorato4be866d2010-10-10 11:26:02 -0700152 private int mDragOutlineCurrent = 0;
Patrick Dubroy8e58e912010-10-14 13:21:48 -0700153 private final Paint mDragOutlinePaint = new Paint();
Winson Chung150fbab2010-09-29 17:14:26 -0700154
Sebastian Francod4682992022-10-05 13:03:09 -0500155 @Thunk final ArrayMap<CellLayoutLayoutParams, Animator> mReorderAnimators = new ArrayMap<>();
Adam Cohend9162062020-03-24 16:35:35 -0700156 @Thunk final ArrayMap<Reorderable, ReorderPreviewAnimation> mShakeAnimators = new ArrayMap<>();
Adam Cohen19f37922012-03-21 11:59:11 -0700157
158 private boolean mItemPlacementDirty = false;
Adam Cohenbfbfd262011-06-13 16:55:12 -0700159
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800160 // Used to visualize the grid and drop locations
161 private boolean mVisualizeCells = false;
162 private boolean mVisualizeDropLocation = true;
163 private RectF mVisualizeGridRect = new RectF();
164 private Paint mVisualizeGridPaint = new Paint();
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800165 private int mGridVisualizationRoundingRadius;
166 private float mGridAlpha = 0f;
167 private int mGridColor = 0;
Sebastian Franco09589322022-11-02 15:25:58 -0700168 protected float mSpringLoadedProgress = 0f;
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800169 private float mScrollProgress = 0f;
170
Patrick Dubroy6569f2c2010-07-12 14:25:18 -0700171 // When a drag operation is in progress, holds the nearest cell to the touch point
172 private final int[] mDragCell = new int[2];
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800173 private final int[] mDragCellSpan = new int[2];
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800174
Joe Onorato4be866d2010-10-10 11:26:02 -0700175 private boolean mDragging = false;
176
Rajeev Kumar9962dbe2017-06-12 12:16:20 -0700177 private final TimeInterpolator mEaseOutInterpolator;
Sebastian Franco09589322022-11-02 15:25:58 -0700178 protected final ShortcutAndWidgetContainer mShortcutsAndWidgets;
Sebastian Franco9ea36d42023-09-21 13:56:42 -0700179 @Px
180 protected int mSpaceBetweenCellLayoutsPx = 0;
Patrick Dubroyce34a972010-10-19 10:34:32 -0700181
Sunny Goyalc13403c2016-11-18 23:44:48 -0800182 @Retention(RetentionPolicy.SOURCE)
183 @IntDef({WORKSPACE, HOTSEAT, FOLDER})
184 public @interface ContainerType{}
185 public static final int WORKSPACE = 0;
186 public static final int HOTSEAT = 1;
187 public static final int FOLDER = 2;
188
189 @ContainerType private final int mContainerType;
190
Jon Mirandab28c4fc2017-06-20 10:58:36 -0700191 private final float mChildScale = 1f;
Andrew Flynn0dca1ec2012-02-29 13:33:22 -0800192
Adam Cohenfa3c58f2013-12-06 16:10:55 -0800193 public static final int MODE_SHOW_REORDER_HINT = 0;
194 public static final int MODE_DRAG_OVER = 1;
195 public static final int MODE_ON_DROP = 2;
196 public static final int MODE_ON_DROP_EXTERNAL = 3;
197 public static final int MODE_ACCEPT_DROP = 4;
Adam Cohen19f37922012-03-21 11:59:11 -0700198 private static final boolean DESTRUCTIVE_REORDER = false;
Adam Cohen482ed822012-03-02 14:15:13 -0800199 private static final boolean DEBUG_VISUALIZE_OCCUPIED = false;
200
Adam Cohenfa3c58f2013-12-06 16:10:55 -0800201 private static final float REORDER_PREVIEW_MAGNITUDE = 0.12f;
Adam Cohen19f37922012-03-21 11:59:11 -0700202 private static final int REORDER_ANIMATION_DURATION = 150;
Sunny Goyalc13403c2016-11-18 23:44:48 -0800203 @Thunk final float mReorderPreviewAnimationMagnitude;
Adam Cohen19f37922012-03-21 11:59:11 -0700204
Rajeev Kumar9962dbe2017-06-12 12:16:20 -0700205 private final ArrayList<View> mIntersectingViews = new ArrayList<>();
206 private final Rect mOccupiedRect = new Rect();
Sebastian Francoe4c03452022-12-27 14:50:02 -0600207 public final int[] mDirectionVector = new int[2];
Steven Ng30dd1d62021-03-15 21:45:49 +0000208
Sebastian Franco53a15a42022-10-25 17:28:54 -0700209 ItemConfiguration mPreviousSolution = null;
Adam Cohenb209e632012-03-27 17:09:36 -0700210 private static final int INVALID_DIRECTION = -100;
Adam Cohen482ed822012-03-02 14:15:13 -0800211
Sunny Goyal2805e632015-05-20 15:35:32 -0700212 private final Rect mTempRect = new Rect();
Jonathan Miranda21930da2021-05-03 18:44:13 +0000213 private final RectF mTempRectF = new RectF();
Jon Miranda88b7f6a2021-05-03 16:49:53 -0700214 private final float[] mTmpFloatArray = new float[4];
Winson Chung3a6e7f32013-10-09 15:50:52 -0700215
Sunny Goyal73b5a272019-12-09 14:55:56 -0800216 private static final Paint sPaint = new Paint();
Romain Guy8a0bff52012-05-06 13:14:33 -0700217
Adam Cohenc9735cf2015-01-23 16:11:55 -0800218 // Related to accessible drag and drop
Adam Cohen6e7c37a2020-06-25 19:22:37 -0700219 DragAndDropAccessibilityDelegate mTouchHelper;
Adam Cohenc9735cf2015-01-23 16:11:55 -0800220
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800221
222 public static final FloatProperty<CellLayout> SPRING_LOADED_PROGRESS =
223 new FloatProperty<CellLayout>("spring_loaded_progress") {
224 @Override
225 public Float get(CellLayout cl) {
226 return cl.getSpringLoadedProgress();
227 }
228
229 @Override
230 public void setValue(CellLayout cl, float progress) {
231 cl.setSpringLoadedProgress(progress);
232 }
233 };
234
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800235 public CellLayout(Context context) {
236 this(context, null);
237 }
238
239 public CellLayout(Context context, AttributeSet attrs) {
240 this(context, attrs, 0);
241 }
242
243 public CellLayout(Context context, AttributeSet attrs, int defStyle) {
244 super(context, attrs, defStyle);
Sunny Goyalc13403c2016-11-18 23:44:48 -0800245 TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.CellLayout, defStyle, 0);
246 mContainerType = a.getInteger(R.styleable.CellLayout_containerType, WORKSPACE);
247 a.recycle();
Patrick Dubroy6569f2c2010-07-12 14:25:18 -0700248
249 // A ViewGroup usually does not draw, but CellLayout needs to draw a rectangle to show
250 // the user where a dragged item will land when dropped.
251 setWillNotDraw(false);
Romain Guyce3cbd12013-02-25 15:00:36 -0800252 setClipToPadding(false);
Pat Manningd0f729d2023-01-09 12:04:25 +0000253 setClipChildren(false);
Sunny Goyalab770a12018-11-14 15:17:26 -0800254 mActivity = ActivityContext.lookupContext(context);
Steven Ngcc505b82021-03-18 23:04:35 +0000255 DeviceProfile deviceProfile = mActivity.getDeviceProfile();
Michael Jurkaa63c4522010-08-19 13:52:27 -0700256
Thales Lima8cd020b2022-03-15 20:15:14 +0000257 resetCellSizeInternal(deviceProfile);
Sunny Goyalff4ba2d2016-04-02 14:12:34 -0700258
Steven Ngcc505b82021-03-18 23:04:35 +0000259 mCountX = deviceProfile.inv.numColumns;
260 mCountY = deviceProfile.inv.numRows;
Sunny Goyalff4ba2d2016-04-02 14:12:34 -0700261 mOccupied = new GridOccupancy(mCountX, mCountY);
262 mTmpOccupied = new GridOccupancy(mCountX, mCountY);
263
Samuel Fufa1e2d0042019-11-18 17:12:46 -0800264 mFolderLeaveBehind.mDelegateCellX = -1;
265 mFolderLeaveBehind.mDelegateCellY = -1;
Adam Cohenefca0272016-02-24 19:19:06 -0800266
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800267 setAlwaysDrawnWithCacheEnabled(false);
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700268
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800269 Resources res = getResources();
270
271 mBackground = getContext().getDrawable(R.drawable.bg_celllayout);
Sunny Goyal2805e632015-05-20 15:35:32 -0700272 mBackground.setCallback(this);
Sunny Goyalaeb16432017-10-16 11:46:41 -0700273 mBackground.setAlpha(0);
Michael Jurka33945b22010-12-21 18:19:38 -0800274
Yogisha Dixitc0ac1dd2021-05-29 00:26:25 +0100275 mGridColor = Themes.getAttrColor(getContext(), R.attr.workspaceAccentColor);
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800276 mGridVisualizationRoundingRadius =
277 res.getDimensionPixelSize(R.dimen.grid_visualization_rounding_radius);
Steven Ngcc505b82021-03-18 23:04:35 +0000278 mReorderPreviewAnimationMagnitude = (REORDER_PREVIEW_MAGNITUDE * deviceProfile.iconSizePx);
Adam Cohen19f37922012-03-21 11:59:11 -0700279
Patrick Dubroy046e7eb2010-10-06 12:14:43 -0700280 // Initialize the data structures used for the drag visualization.
Kateryna Ivanova71203732023-05-24 15:09:00 +0000281 mEaseOutInterpolator = Interpolators.DECELERATE_QUINT; // Quint ease out
Winson Chungb8c69f32011-10-19 21:36:08 -0700282 mDragCell[0] = mDragCell[1] = -1;
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800283 mDragCellSpan[0] = mDragCellSpan[1] = -1;
Joe Onorato4be866d2010-10-10 11:26:02 -0700284 for (int i = 0; i < mDragOutlines.length; i++) {
Sunny Goyal669b71f2023-01-27 14:37:07 -0800285 mDragOutlines[i] = new CellLayoutLayoutParams(0, 0, 0, 0);
Patrick Dubroy046e7eb2010-10-06 12:14:43 -0700286 }
Mario Bertschler54ba6012017-06-08 10:53:53 -0700287 mDragOutlinePaint.setColor(Themes.getAttrColor(context, R.attr.workspaceTextColor));
Patrick Dubroy046e7eb2010-10-06 12:14:43 -0700288
289 // When dragging things around the home screens, we show a green outline of
290 // where the item will land. The outlines gradually fade out, leaving a trail
291 // behind the drag path.
292 // Set up all the animations that are used to implement this fading.
293 final int duration = res.getInteger(R.integer.config_dragOutlineFadeTime);
Chet Haase472b2812010-10-14 07:02:04 -0700294 final float fromAlphaValue = 0;
295 final float toAlphaValue = (float)res.getInteger(R.integer.config_dragOutlineMaxAlpha);
Joe Onorato4be866d2010-10-10 11:26:02 -0700296
Patrick Dubroy8e58e912010-10-14 13:21:48 -0700297 Arrays.fill(mDragOutlineAlphas, fromAlphaValue);
Joe Onorato4be866d2010-10-10 11:26:02 -0700298
299 for (int i = 0; i < mDragOutlineAnims.length; i++) {
Patrick Dubroy046e7eb2010-10-06 12:14:43 -0700300 final InterruptibleInOutAnimator anim =
Sebastian Francof153d912022-04-22 16:15:27 -0500301 new InterruptibleInOutAnimator(duration, fromAlphaValue, toAlphaValue);
Patrick Dubroyce34a972010-10-19 10:34:32 -0700302 anim.getAnimator().setInterpolator(mEaseOutInterpolator);
Patrick Dubroy046e7eb2010-10-06 12:14:43 -0700303 final int thisIndex = i;
Chet Haase472b2812010-10-14 07:02:04 -0700304 anim.getAnimator().addUpdateListener(new AnimatorUpdateListener() {
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700305 public void onAnimationUpdate(ValueAnimator animation) {
Joe Onorato4be866d2010-10-10 11:26:02 -0700306 // If an animation is started and then stopped very quickly, we can still
307 // get spurious updates we've cleared the tag. Guard against this.
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800308 mDragOutlineAlphas[thisIndex] = (Float) animation.getAnimatedValue();
309 CellLayout.this.invalidate();
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700310 }
311 });
Joe Onorato4be866d2010-10-10 11:26:02 -0700312 // The animation holds a reference to the drag outline bitmap as long is it's
313 // running. This way the bitmap can be GCed when the animations are complete.
Joe Onorato4be866d2010-10-10 11:26:02 -0700314 mDragOutlineAnims[i] = anim;
Patrick Dubroyde7658b2010-09-27 11:15:43 -0700315 }
Patrick Dubroyce34a972010-10-19 10:34:32 -0700316
Sunny Goyalc13403c2016-11-18 23:44:48 -0800317 mShortcutsAndWidgets = new ShortcutAndWidgetContainer(context, mContainerType);
Jon Miranda228877d2021-02-09 11:05:00 -0500318 mShortcutsAndWidgets.setCellDimensions(mCellWidth, mCellHeight, mCountX, mCountY,
Thales Lima78d00ad2021-09-30 11:29:06 +0100319 mBorderSpace);
Michael Jurkaa52570f2012-03-20 03:18:20 -0700320 addView(mShortcutsAndWidgets);
Michael Jurka18014792010-10-14 09:01:34 -0700321 }
322
Sunny Goyal9b180102020-03-11 10:02:29 -0700323 /**
324 * Sets or clears a delegate used for accessible drag and drop
325 */
326 public void setDragAndDropAccessibilityDelegate(DragAndDropAccessibilityDelegate delegate) {
327 setOnClickListener(delegate);
Sunny Goyal9b180102020-03-11 10:02:29 -0700328 ViewCompat.setAccessibilityDelegate(this, delegate);
329
Adam Cohen6e7c37a2020-06-25 19:22:37 -0700330 mTouchHelper = delegate;
331 int accessibilityFlag = mTouchHelper != null
Sunny Goyal9b180102020-03-11 10:02:29 -0700332 ? IMPORTANT_FOR_ACCESSIBILITY_YES : IMPORTANT_FOR_ACCESSIBILITY_NO;
333 setImportantForAccessibility(accessibilityFlag);
334 getShortcutsAndWidgets().setImportantForAccessibility(accessibilityFlag);
Adam Cohenc9735cf2015-01-23 16:11:55 -0800335
Sunny Goyal384b5782021-02-09 22:50:02 -0800336 // ExploreByTouchHelper sets focusability. Clear it when the delegate is cleared.
337 setFocusable(delegate != null);
Adam Cohenc9735cf2015-01-23 16:11:55 -0800338 // Invalidate the accessibility hierarchy
339 if (getParent() != null) {
340 getParent().notifySubtreeAccessibilityStateChanged(
341 this, this, AccessibilityEvent.CONTENT_CHANGE_TYPE_SUBTREE);
342 }
343 }
344
Sunny Goyala4647b62021-02-02 13:45:34 -0800345 /**
346 * Returns the currently set accessibility delegate
347 */
348 public DragAndDropAccessibilityDelegate getDragAndDropAccessibilityDelegate() {
349 return mTouchHelper;
350 }
351
Adam Cohenc9735cf2015-01-23 16:11:55 -0800352 @Override
Adam Cohen6e7c37a2020-06-25 19:22:37 -0700353 public boolean dispatchHoverEvent(MotionEvent event) {
354 // Always attempt to dispatch hover events to accessibility first.
355 if (mTouchHelper != null && mTouchHelper.dispatchHoverEvent(event)) {
356 return true;
357 }
358 return super.dispatchHoverEvent(event);
359 }
360
361 @Override
Adam Cohenc9735cf2015-01-23 16:11:55 -0800362 public boolean onInterceptTouchEvent(MotionEvent ev) {
Winson Chungf9935182020-10-23 09:26:44 -0700363 return mTouchHelper != null
364 || (mInterceptTouchListener != null && mInterceptTouchListener.onTouch(this, ev));
Adam Cohenc9735cf2015-01-23 16:11:55 -0800365 }
366
Chris Craik01f2d7f2013-10-01 14:41:56 -0700367 public void enableHardwareLayer(boolean hasLayer) {
368 mShortcutsAndWidgets.setLayerType(hasLayer ? LAYER_TYPE_HARDWARE : LAYER_TYPE_NONE, sPaint);
Michael Jurkad51f33a2012-06-28 15:35:26 -0700369 }
370
vadimt04f356f2019-02-14 18:46:36 -0800371 public boolean isHardwareLayerEnabled() {
372 return mShortcutsAndWidgets.getLayerType() == LAYER_TYPE_HARDWARE;
373 }
374
Thales Lima8cd020b2022-03-15 20:15:14 +0000375 /**
376 * Change sizes of cells
377 *
378 * @param width the new width of the cells
379 * @param height the new height of the cells
380 */
Winson Chung5f8afe62013-08-12 16:19:28 -0700381 public void setCellDimensions(int width, int height) {
Winson Chung11a1a532013-09-13 11:14:45 -0700382 mFixedCellWidth = mCellWidth = width;
383 mFixedCellHeight = mCellHeight = height;
Jon Miranda228877d2021-02-09 11:05:00 -0500384 mShortcutsAndWidgets.setCellDimensions(mCellWidth, mCellHeight, mCountX, mCountY,
Thales Lima78d00ad2021-09-30 11:29:06 +0100385 mBorderSpace);
Winson Chung5f8afe62013-08-12 16:19:28 -0700386 }
387
Thales Lima8cd020b2022-03-15 20:15:14 +0000388 private void resetCellSizeInternal(DeviceProfile deviceProfile) {
389 switch (mContainerType) {
390 case FOLDER:
Jordan Silva637f4eb2023-06-13 11:21:53 +0100391 mBorderSpace = new Point(deviceProfile.folderCellLayoutBorderSpacePx);
Thales Lima8cd020b2022-03-15 20:15:14 +0000392 break;
393 case HOTSEAT:
394 mBorderSpace = new Point(deviceProfile.hotseatBorderSpace,
395 deviceProfile.hotseatBorderSpace);
396 break;
397 case WORKSPACE:
398 default:
399 mBorderSpace = new Point(deviceProfile.cellLayoutBorderSpacePx);
400 break;
401 }
402
403 mCellWidth = mCellHeight = -1;
404 mFixedCellWidth = mFixedCellHeight = -1;
405 }
406
407 /**
408 * Reset the cell sizes and border space
409 */
410 public void resetCellSize(DeviceProfile deviceProfile) {
411 resetCellSizeInternal(deviceProfile);
412 requestLayout();
413 }
414
Adam Cohen2801caf2011-05-13 20:57:39 -0700415 public void setGridSize(int x, int y) {
416 mCountX = x;
417 mCountY = y;
Sunny Goyalff4ba2d2016-04-02 14:12:34 -0700418 mOccupied = new GridOccupancy(mCountX, mCountY);
419 mTmpOccupied = new GridOccupancy(mCountX, mCountY);
Jon Miranda228877d2021-02-09 11:05:00 -0500420 mShortcutsAndWidgets.setCellDimensions(mCellWidth, mCellHeight, mCountX, mCountY,
Thales Lima78d00ad2021-09-30 11:29:06 +0100421 mBorderSpace);
Adam Cohen76fc0852011-06-17 13:26:23 -0700422 requestLayout();
Adam Cohen2801caf2011-05-13 20:57:39 -0700423 }
424
Adam Cohen2374abf2013-04-16 14:56:57 -0700425 // Set whether or not to invert the layout horizontally if the layout is in RTL mode.
426 public void setInvertIfRtl(boolean invert) {
427 mShortcutsAndWidgets.setInvertIfRtl(invert);
428 }
429
Adam Cohen917e3882013-10-31 15:03:35 -0700430 public void setDropPending(boolean pending) {
431 mDropPending = pending;
432 }
433
434 public boolean isDropPending() {
435 return mDropPending;
436 }
437
Adam Cohenc50438c2014-08-19 17:43:05 -0700438 void setIsDragOverlapping(boolean isDragOverlapping) {
439 if (mIsDragOverlapping != isDragOverlapping) {
440 mIsDragOverlapping = isDragOverlapping;
Sunny Goyalf5440cb2016-12-14 15:13:00 -0800441 mBackground.setState(mIsDragOverlapping
442 ? BACKGROUND_STATE_ACTIVE : BACKGROUND_STATE_DEFAULT);
Adam Cohenc50438c2014-08-19 17:43:05 -0700443 invalidate();
444 }
445 }
446
Sunny Goyale2fd14b2015-08-27 17:45:46 -0700447 @Override
448 protected void dispatchSaveInstanceState(SparseArray<Parcelable> container) {
Sunny Goyal7ce471b2017-08-02 03:37:39 -0700449 ParcelableSparseArray jail = getJailedArray(container);
450 super.dispatchSaveInstanceState(jail);
451 container.put(R.id.cell_layout_jail_id, jail);
Sunny Goyale2fd14b2015-08-27 17:45:46 -0700452 }
453
454 @Override
455 protected void dispatchRestoreInstanceState(SparseArray<Parcelable> container) {
Sunny Goyal7ce471b2017-08-02 03:37:39 -0700456 super.dispatchRestoreInstanceState(getJailedArray(container));
Sunny Goyale2fd14b2015-08-27 17:45:46 -0700457 }
458
Sunny Goyal7ce471b2017-08-02 03:37:39 -0700459 /**
460 * Wrap the SparseArray in another Parcelable so that the item ids do not conflict with our
461 * our internal resource ids
462 */
Sunny Goyale2fd14b2015-08-27 17:45:46 -0700463 private ParcelableSparseArray getJailedArray(SparseArray<Parcelable> container) {
464 final Parcelable parcelable = container.get(R.id.cell_layout_jail_id);
465 return parcelable instanceof ParcelableSparseArray ?
466 (ParcelableSparseArray) parcelable : new ParcelableSparseArray();
467 }
468
Tony Wickham0f97b782015-12-02 17:55:07 -0800469 public boolean getIsDragOverlapping() {
470 return mIsDragOverlapping;
471 }
472
Jeff Sharkey83f111d2009-04-20 21:03:13 -0700473 @Override
Patrick Dubroy1262e362010-10-06 15:49:50 -0700474 protected void onDraw(Canvas canvas) {
Michael Jurka3e7c7632010-10-02 16:01:03 -0700475 // When we're large, we are either drawn in a "hover" state (ie when dragging an item to
476 // a neighboring page) or with just a normal background (if backgroundAlpha > 0.0f)
477 // When we're small, we are either drawn normally or in the "accepts drops" state (during
478 // a drag). However, we also drag the mini hover background *over* one of those two
479 // backgrounds
Sunny Goyalaeb16432017-10-16 11:46:41 -0700480 if (mBackground.getAlpha() > 0) {
Sunny Goyal2805e632015-05-20 15:35:32 -0700481 mBackground.draw(canvas);
Michael Jurkaa63c4522010-08-19 13:52:27 -0700482 }
Romain Guya6abce82009-11-10 02:54:41 -0800483
Adam Cohen482ed822012-03-02 14:15:13 -0800484 if (DEBUG_VISUALIZE_OCCUPIED) {
Tony Wickham0ac045f2021-11-03 13:17:02 -0700485 Rect cellBounds = new Rect();
486 // Will contain the bounds of the cell including spacing between cells.
487 Rect cellBoundsWithSpacing = new Rect();
Tony Wickham12784902021-11-03 14:02:10 -0700488 int[] targetCell = new int[2];
Tony Wickham0ac045f2021-11-03 13:17:02 -0700489 int[] cellCenter = new int[2];
490 Paint debugPaint = new Paint();
491 debugPaint.setStrokeWidth(Utilities.dpToPx(1));
492 for (int x = 0; x < mCountX; x++) {
493 for (int y = 0; y < mCountY; y++) {
494 if (!mOccupied.cells[x][y]) {
495 continue;
Adam Cohen482ed822012-03-02 14:15:13 -0800496 }
Tony Wickham12784902021-11-03 14:02:10 -0700497 targetCell[0] = x;
498 targetCell[1] = y;
Tony Wickham0ac045f2021-11-03 13:17:02 -0700499
Tony Wickham12784902021-11-03 14:02:10 -0700500 boolean canCreateFolder = canCreateFolder(getChildAt(x, y));
Tony Wickham0ac045f2021-11-03 13:17:02 -0700501 cellToRect(x, y, 1, 1, cellBounds);
502 cellBoundsWithSpacing.set(cellBounds);
503 cellBoundsWithSpacing.inset(-mBorderSpace.x / 2, -mBorderSpace.y / 2);
Tony Wickham3cfa5ed2021-11-03 13:20:43 -0700504 getWorkspaceCellVisualCenter(x, y, cellCenter);
Tony Wickham0ac045f2021-11-03 13:17:02 -0700505
506 canvas.save();
507 canvas.clipRect(cellBoundsWithSpacing);
508
509 // Draw reorder drag target.
510 debugPaint.setColor(Color.RED);
Sebastian Franco6e1024e2022-07-29 13:46:49 -0700511 canvas.drawCircle(cellCenter[0], cellCenter[1],
512 getReorderRadius(targetCell, 1, 1), debugPaint);
Tony Wickham0ac045f2021-11-03 13:17:02 -0700513
514 // Draw folder creation drag target.
515 if (canCreateFolder) {
516 debugPaint.setColor(Color.GREEN);
517 canvas.drawCircle(cellCenter[0], cellCenter[1],
Tony Wickham12784902021-11-03 14:02:10 -0700518 getFolderCreationRadius(targetCell), debugPaint);
Tony Wickham0ac045f2021-11-03 13:17:02 -0700519 }
520
521 canvas.restore();
Adam Cohen482ed822012-03-02 14:15:13 -0800522 }
523 }
524 }
525
Samuel Fufa1e2d0042019-11-18 17:12:46 -0800526 for (int i = 0; i < mDelegatedCellDrawings.size(); i++) {
527 DelegatedCellDrawing cellDrawing = mDelegatedCellDrawings.get(i);
528 cellToPoint(cellDrawing.mDelegateCellX, cellDrawing.mDelegateCellY, mTempLocation);
Adam Cohenefca0272016-02-24 19:19:06 -0800529 canvas.save();
530 canvas.translate(mTempLocation[0], mTempLocation[1]);
Samuel Fufa1e2d0042019-11-18 17:12:46 -0800531 cellDrawing.drawUnderItem(canvas);
Adam Cohenefca0272016-02-24 19:19:06 -0800532 canvas.restore();
Adam Cohen69ce2e52011-07-03 19:25:21 -0700533 }
Adam Cohenc51934b2011-07-26 21:07:43 -0700534
Samuel Fufa1e2d0042019-11-18 17:12:46 -0800535 if (mFolderLeaveBehind.mDelegateCellX >= 0 && mFolderLeaveBehind.mDelegateCellY >= 0) {
536 cellToPoint(mFolderLeaveBehind.mDelegateCellX,
537 mFolderLeaveBehind.mDelegateCellY, mTempLocation);
Adam Cohenefca0272016-02-24 19:19:06 -0800538 canvas.save();
539 canvas.translate(mTempLocation[0], mTempLocation[1]);
Tony Wickhamec6fd6f2023-03-11 02:08:57 +0000540 mFolderLeaveBehind.drawLeaveBehind(canvas, FOLDER_LEAVE_BEHIND_COLOR);
Adam Cohenefca0272016-02-24 19:19:06 -0800541 canvas.restore();
Adam Cohenc51934b2011-07-26 21:07:43 -0700542 }
Adam Cohen65086992020-02-19 08:40:49 -0800543
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800544 if (mVisualizeCells || mVisualizeDropLocation) {
Adam Cohen65086992020-02-19 08:40:49 -0800545 visualizeGrid(canvas);
546 }
547 }
548
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800549 /**
Tony Wickham12784902021-11-03 14:02:10 -0700550 * Returns whether dropping an icon on the given View can create (or add to) a folder.
551 */
552 private boolean canCreateFolder(View child) {
553 return child instanceof DraggableView
554 && ((DraggableView) child).getViewType() == DRAGGABLE_ICON;
555 }
556
557 /**
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800558 * Indicates the progress of the Workspace entering the SpringLoaded state; allows the
559 * CellLayout to update various visuals for this state.
560 *
561 * @param progress
562 */
563 public void setSpringLoadedProgress(float progress) {
564 if (Float.compare(progress, mSpringLoadedProgress) != 0) {
565 mSpringLoadedProgress = progress;
fbarone74256b2023-04-10 14:50:31 -0700566 updateBgAlpha();
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800567 setGridAlpha(progress);
568 }
569 }
570
571 /**
572 * See setSpringLoadedProgress
573 * @return progress
574 */
575 public float getSpringLoadedProgress() {
576 return mSpringLoadedProgress;
577 }
578
Sebastian Franco09589322022-11-02 15:25:58 -0700579 protected void updateBgAlpha() {
Charlie Anderson438d4052023-04-25 14:38:57 -0400580 if (!getWorkspace().mLauncher.isInState(EDIT_MODE)) {
581 mBackground.setAlpha((int) (mSpringLoadedProgress * 255));
582 }
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800583 }
584
585 /**
586 * Set the progress of this page's scroll
587 *
588 * @param progress 0 if the screen is centered, +/-1 if it is to the right / left respectively
589 */
590 public void setScrollProgress(float progress) {
591 if (Float.compare(Math.abs(progress), mScrollProgress) != 0) {
592 mScrollProgress = Math.abs(progress);
fbarone74256b2023-04-10 14:50:31 -0700593 updateBgAlpha();
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800594 }
595 }
596
597 private void setGridAlpha(float gridAlpha) {
598 if (Float.compare(gridAlpha, mGridAlpha) != 0) {
599 mGridAlpha = gridAlpha;
600 invalidate();
601 }
602 }
603
Adam Cohen65086992020-02-19 08:40:49 -0800604 protected void visualizeGrid(Canvas canvas) {
Adam Cohen0c4d2782021-04-29 15:56:13 -0700605 DeviceProfile dp = mActivity.getDeviceProfile();
Alex Chau51da2192022-05-20 13:32:10 +0100606 int paddingX = Math.min((mCellWidth - dp.iconSizePx) / 2, dp.gridVisualizationPaddingX);
607 int paddingY = Math.min((mCellHeight - dp.iconSizePx) / 2, dp.gridVisualizationPaddingY);
Adam Cohen65086992020-02-19 08:40:49 -0800608
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800609 mVisualizeGridPaint.setStrokeWidth(8);
Adam Cohen65086992020-02-19 08:40:49 -0800610
Sebastian Franco0dd5db82023-10-13 11:09:21 -0700611 // This is used for debugging purposes only
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800612 if (mVisualizeCells) {
Sebastian Franco0dd5db82023-10-13 11:09:21 -0700613 int paintAlpha = (int) (120 * mGridAlpha);
614 mVisualizeGridPaint.setColor(ColorUtils.setAlphaComponent(mGridColor, paintAlpha));
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800615 for (int i = 0; i < mCountX; i++) {
616 for (int j = 0; j < mCountY; j++) {
Sebastian Franco0dd5db82023-10-13 11:09:21 -0700617 cellToRect(i, j, 1, 1, mTempOnDrawCellToRect);
618 mVisualizeGridRect.set(mTempOnDrawCellToRect);
619 mVisualizeGridRect.inset(paddingX, paddingY);
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800620 mVisualizeGridPaint.setStyle(Paint.Style.FILL);
621 canvas.drawRoundRect(mVisualizeGridRect, mGridVisualizationRoundingRadius,
622 mGridVisualizationRoundingRadius, mVisualizeGridPaint);
623 }
624 }
625 }
Adam Cohen65086992020-02-19 08:40:49 -0800626
fbarone74256b2023-04-10 14:50:31 -0700627 if (mVisualizeDropLocation) {
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800628 for (int i = 0; i < mDragOutlines.length; i++) {
629 final float alpha = mDragOutlineAlphas[i];
630 if (alpha <= 0) continue;
Sebastian Franco0dd5db82023-10-13 11:09:21 -0700631 CellLayoutLayoutParams params = mDragOutlines[i];
632 cellToRect(params.getCellX(), params.getCellY(), params.cellHSpan, params.cellVSpan,
633 mTempOnDrawCellToRect);
634 mVisualizeGridRect.set(mTempOnDrawCellToRect);
635 mVisualizeGridRect.inset(paddingX, paddingY);
Adam Cohen65086992020-02-19 08:40:49 -0800636
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800637 mVisualizeGridPaint.setAlpha(255);
Adam Cohen65086992020-02-19 08:40:49 -0800638 mVisualizeGridPaint.setStyle(Paint.Style.STROKE);
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800639 mVisualizeGridPaint.setColor(Color.argb((int) (alpha),
640 Color.red(mGridColor), Color.green(mGridColor), Color.blue(mGridColor)));
Adam Cohen65086992020-02-19 08:40:49 -0800641
Sebastian Franco9ea36d42023-09-21 13:56:42 -0700642 canvas.save();
643 canvas.translate(getMarginForGivenCellParams(params), 0);
Adam Cohenf7ca3b42021-02-22 11:03:58 -0800644 canvas.drawRoundRect(mVisualizeGridRect, mGridVisualizationRoundingRadius,
645 mGridVisualizationRoundingRadius, mVisualizeGridPaint);
Sebastian Franco9ea36d42023-09-21 13:56:42 -0700646 canvas.restore();
Adam Cohen65086992020-02-19 08:40:49 -0800647 }
648 }
Adam Cohen69ce2e52011-07-03 19:25:21 -0700649 }
650
Sebastian Franco9ea36d42023-09-21 13:56:42 -0700651 protected float getMarginForGivenCellParams(CellLayoutLayoutParams params) {
652 return 0;
653 }
654
Adam Cohenefca0272016-02-24 19:19:06 -0800655 @Override
656 protected void dispatchDraw(Canvas canvas) {
657 super.dispatchDraw(canvas);
658
Samuel Fufa1e2d0042019-11-18 17:12:46 -0800659 for (int i = 0; i < mDelegatedCellDrawings.size(); i++) {
660 DelegatedCellDrawing bg = mDelegatedCellDrawings.get(i);
661 cellToPoint(bg.mDelegateCellX, bg.mDelegateCellY, mTempLocation);
662 canvas.save();
663 canvas.translate(mTempLocation[0], mTempLocation[1]);
664 bg.drawOverItem(canvas);
665 canvas.restore();
Adam Cohenefca0272016-02-24 19:19:06 -0800666 }
Adam Cohen69ce2e52011-07-03 19:25:21 -0700667 }
668
Samuel Fufa1e2d0042019-11-18 17:12:46 -0800669 /**
670 * Add Delegated cell drawing
671 */
672 public void addDelegatedCellDrawing(DelegatedCellDrawing bg) {
673 mDelegatedCellDrawings.add(bg);
Adam Cohenefca0272016-02-24 19:19:06 -0800674 }
Samuel Fufa1e2d0042019-11-18 17:12:46 -0800675
676 /**
677 * Remove item from DelegatedCellDrawings
678 */
679 public void removeDelegatedCellDrawing(DelegatedCellDrawing bg) {
680 mDelegatedCellDrawings.remove(bg);
Patrick Dubroy6569f2c2010-07-12 14:25:18 -0700681 }
682
Adam Cohenc51934b2011-07-26 21:07:43 -0700683 public void setFolderLeaveBehindCell(int x, int y) {
Adam Cohenefca0272016-02-24 19:19:06 -0800684 View child = getChildAt(x, y);
Sunny Goyalab770a12018-11-14 15:17:26 -0800685 mFolderLeaveBehind.setup(getContext(), mActivity, null,
Adam Cohenefca0272016-02-24 19:19:06 -0800686 child.getMeasuredWidth(), child.getPaddingTop());
687
Samuel Fufa1e2d0042019-11-18 17:12:46 -0800688 mFolderLeaveBehind.mDelegateCellX = x;
689 mFolderLeaveBehind.mDelegateCellY = y;
Adam Cohenc51934b2011-07-26 21:07:43 -0700690 invalidate();
691 }
692
693 public void clearFolderLeaveBehind() {
Samuel Fufa1e2d0042019-11-18 17:12:46 -0800694 mFolderLeaveBehind.mDelegateCellX = -1;
695 mFolderLeaveBehind.mDelegateCellY = -1;
Adam Cohenc51934b2011-07-26 21:07:43 -0700696 invalidate();
697 }
698
Patrick Dubroy6569f2c2010-07-12 14:25:18 -0700699 @Override
Michael Jurkae6235dd2011-10-04 15:02:05 -0700700 public boolean shouldDelayChildPressedState() {
701 return false;
702 }
703
Adam Cohen1462de32012-07-24 22:34:36 -0700704 public void restoreInstanceState(SparseArray<Parcelable> states) {
Sunny Goyal33a152f2014-07-22 12:13:14 -0700705 try {
706 dispatchRestoreInstanceState(states);
707 } catch (IllegalArgumentException ex) {
Zak Cohen3eeb41d2020-02-14 14:15:13 -0800708 if (FeatureFlags.IS_STUDIO_BUILD) {
Sunny Goyal33a152f2014-07-22 12:13:14 -0700709 throw ex;
710 }
711 // Mismatched viewId / viewType preventing restore. Skip restore on production builds.
712 Log.e(TAG, "Ignoring an error while restoring a view instance state", ex);
713 }
Adam Cohen1462de32012-07-24 22:34:36 -0700714 }
715
Michael Jurkae6235dd2011-10-04 15:02:05 -0700716 @Override
Jeff Sharkey83f111d2009-04-20 21:03:13 -0700717 public void cancelLongPress() {
718 super.cancelLongPress();
719
720 // Cancel long press for all children
721 final int count = getChildCount();
722 for (int i = 0; i < count; i++) {
723 final View child = getChildAt(i);
724 child.cancelLongPress();
725 }
726 }
727
Michael Jurkadee05892010-07-27 10:01:56 -0700728 public void setOnInterceptTouchListener(View.OnTouchListener listener) {
729 mInterceptTouchListener = listener;
730 }
731
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800732 public int getCountX() {
Adam Cohend22015c2010-07-26 22:02:18 -0700733 return mCountX;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800734 }
735
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800736 public int getCountY() {
Adam Cohend22015c2010-07-26 22:02:18 -0700737 return mCountY;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800738 }
739
Sunny Goyalc13403c2016-11-18 23:44:48 -0800740 public boolean acceptsWidget() {
741 return mContainerType == WORKSPACE;
Sunny Goyale9b651e2015-04-24 11:44:51 -0700742 }
743
Sebastian Francod4682992022-10-05 13:03:09 -0500744 /**
745 * Adds the given view to the CellLayout
746 *
747 * @param child view to add.
748 * @param index index of the CellLayout children where to add the view.
749 * @param childId id of the view.
750 * @param params represent the logic of the view on the CellLayout.
751 * @param markCells if the occupied cells should be marked or not
752 * @return if adding the view was successful
753 */
754 public boolean addViewToCellLayout(View child, int index, int childId,
755 CellLayoutLayoutParams params, boolean markCells) {
756 final CellLayoutLayoutParams lp = params;
Winson Chungaafa03c2010-06-11 17:34:16 -0700757
Andrew Flynnde38e422012-05-08 11:22:15 -0700758 // Hotseat icons - remove text
Andrew Flynn0dca1ec2012-02-29 13:33:22 -0800759 if (child instanceof BubbleTextView) {
760 BubbleTextView bubbleChild = (BubbleTextView) child;
Jon Mirandaf1eae802017-10-04 11:23:33 -0700761 bubbleChild.setTextVisibility(mContainerType != HOTSEAT);
Andrew Flynn0dca1ec2012-02-29 13:33:22 -0800762 }
763
Sunny Goyalc13403c2016-11-18 23:44:48 -0800764 child.setScaleX(mChildScale);
765 child.setScaleY(mChildScale);
Adam Cohen307fe232012-08-16 17:55:58 -0700766
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800767 // Generate an id for each view, this assumes we have at most 256x256 cells
768 // per workspace screen
Sebastian Franco877088e2023-01-03 15:16:22 -0700769 if (lp.getCellX() >= 0 && lp.getCellX() <= mCountX - 1
770 && lp.getCellY() >= 0 && lp.getCellY() <= mCountY - 1) {
Winson Chungaafa03c2010-06-11 17:34:16 -0700771 // If the horizontal or vertical span is set to -1, it is taken to
772 // mean that it spans the extent of the CellLayout
Adam Cohend22015c2010-07-26 22:02:18 -0700773 if (lp.cellHSpan < 0) lp.cellHSpan = mCountX;
774 if (lp.cellVSpan < 0) lp.cellVSpan = mCountY;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800775
Winson Chungaafa03c2010-06-11 17:34:16 -0700776 child.setId(childId);
Tony Wickhama0628cc2015-10-14 15:23:04 -0700777 if (LOGD) {
778 Log.d(TAG, "Adding view to ShortcutsAndWidgetsContainer: " + child);
779 }
Sunny Goyalf7a29e82015-04-24 15:20:43 -0700780 mShortcutsAndWidgets.addView(child, index, lp);
Michael Jurkadee05892010-07-27 10:01:56 -0700781
Michael Jurkaf3ca3ab2010-10-20 17:08:24 -0700782 if (markCells) markCellsAsOccupiedForView(child);
Michael Jurka0280c3b2010-09-17 15:00:07 -0700783
Winson Chungaafa03c2010-06-11 17:34:16 -0700784 return true;
785 }
786 return false;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800787 }
Michael Jurka3e7c7632010-10-02 16:01:03 -0700788
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800789 @Override
Michael Jurka0280c3b2010-09-17 15:00:07 -0700790 public void removeAllViews() {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -0700791 mOccupied.clear();
Michael Jurkaa52570f2012-03-20 03:18:20 -0700792 mShortcutsAndWidgets.removeAllViews();
Michael Jurka0280c3b2010-09-17 15:00:07 -0700793 }
794
795 @Override
796 public void removeAllViewsInLayout() {
Michael Jurkaa52570f2012-03-20 03:18:20 -0700797 if (mShortcutsAndWidgets.getChildCount() > 0) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -0700798 mOccupied.clear();
Michael Jurkaa52570f2012-03-20 03:18:20 -0700799 mShortcutsAndWidgets.removeAllViewsInLayout();
Michael Jurka7cfc2822011-08-02 20:19:24 -0700800 }
Michael Jurka0280c3b2010-09-17 15:00:07 -0700801 }
802
803 @Override
804 public void removeView(View view) {
805 markCellsAsUnoccupiedForView(view);
Michael Jurkaa52570f2012-03-20 03:18:20 -0700806 mShortcutsAndWidgets.removeView(view);
Michael Jurka0280c3b2010-09-17 15:00:07 -0700807 }
808
809 @Override
810 public void removeViewAt(int index) {
Michael Jurkaa52570f2012-03-20 03:18:20 -0700811 markCellsAsUnoccupiedForView(mShortcutsAndWidgets.getChildAt(index));
812 mShortcutsAndWidgets.removeViewAt(index);
Michael Jurka0280c3b2010-09-17 15:00:07 -0700813 }
814
815 @Override
816 public void removeViewInLayout(View view) {
817 markCellsAsUnoccupiedForView(view);
Michael Jurkaa52570f2012-03-20 03:18:20 -0700818 mShortcutsAndWidgets.removeViewInLayout(view);
Michael Jurka0280c3b2010-09-17 15:00:07 -0700819 }
820
821 @Override
822 public void removeViews(int start, int count) {
823 for (int i = start; i < start + count; i++) {
Michael Jurkaa52570f2012-03-20 03:18:20 -0700824 markCellsAsUnoccupiedForView(mShortcutsAndWidgets.getChildAt(i));
Michael Jurka0280c3b2010-09-17 15:00:07 -0700825 }
Michael Jurkaa52570f2012-03-20 03:18:20 -0700826 mShortcutsAndWidgets.removeViews(start, count);
Michael Jurka0280c3b2010-09-17 15:00:07 -0700827 }
828
829 @Override
830 public void removeViewsInLayout(int start, int count) {
831 for (int i = start; i < start + count; i++) {
Michael Jurkaa52570f2012-03-20 03:18:20 -0700832 markCellsAsUnoccupiedForView(mShortcutsAndWidgets.getChildAt(i));
Michael Jurka0280c3b2010-09-17 15:00:07 -0700833 }
Michael Jurkaa52570f2012-03-20 03:18:20 -0700834 mShortcutsAndWidgets.removeViewsInLayout(start, count);
Michael Jurkaabded662011-03-04 12:06:57 -0800835 }
836
Patrick Dubroy6569f2c2010-07-12 14:25:18 -0700837 /**
Winson Chungaafa03c2010-06-11 17:34:16 -0700838 * Given a point, return the cell that strictly encloses that point
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800839 * @param x X coordinate of the point
840 * @param y Y coordinate of the point
841 * @param result Array of 2 ints to hold the x and y coordinate of the cell
842 */
Sunny Goyale9b651e2015-04-24 11:44:51 -0700843 public void pointToCellExact(int x, int y, int[] result) {
Winson Chung4b825dcd2011-06-19 12:41:22 -0700844 final int hStartPadding = getPaddingLeft();
845 final int vStartPadding = getPaddingTop();
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800846
Sebastian Francob57c0b22022-06-28 13:54:35 -0700847 result[0] = (x - hStartPadding) / (mCellWidth + mBorderSpace.x);
848 result[1] = (y - vStartPadding) / (mCellHeight + mBorderSpace.y);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800849
Adam Cohend22015c2010-07-26 22:02:18 -0700850 final int xAxis = mCountX;
851 final int yAxis = mCountY;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800852
853 if (result[0] < 0) result[0] = 0;
854 if (result[0] >= xAxis) result[0] = xAxis - 1;
855 if (result[1] < 0) result[1] = 0;
856 if (result[1] >= yAxis) result[1] = yAxis - 1;
857 }
Winson Chungaafa03c2010-06-11 17:34:16 -0700858
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800859 /**
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800860 * Given a cell coordinate, return the point that represents the upper left corner of that cell
Winson Chungaafa03c2010-06-11 17:34:16 -0700861 *
862 * @param cellX X coordinate of the cell
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800863 * @param cellY Y coordinate of the cell
Winson Chungaafa03c2010-06-11 17:34:16 -0700864 *
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800865 * @param result Array of 2 ints to hold the x and y coordinate of the point
866 */
867 void cellToPoint(int cellX, int cellY, int[] result) {
Jon Miranda228877d2021-02-09 11:05:00 -0500868 cellToRect(cellX, cellY, 1, 1, mTempRect);
869 result[0] = mTempRect.left;
870 result[1] = mTempRect.top;
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800871 }
872
Adam Cohene3e27a82011-04-15 12:07:39 -0700873 /**
Adam Cohen482ed822012-03-02 14:15:13 -0800874 * Given a cell coordinate, return the point that represents the center of the cell
Adam Cohene3e27a82011-04-15 12:07:39 -0700875 *
876 * @param cellX X coordinate of the cell
877 * @param cellY Y coordinate of the cell
878 *
879 * @param result Array of 2 ints to hold the x and y coordinate of the point
880 */
881 void cellToCenterPoint(int cellX, int cellY, int[] result) {
Adam Cohen47a876d2012-03-19 13:21:41 -0700882 regionToCenterPoint(cellX, cellY, 1, 1, result);
883 }
884
885 /**
Tony Wickham0ac045f2021-11-03 13:17:02 -0700886 * Given a cell coordinate and span return the point that represents the center of the region
Adam Cohen47a876d2012-03-19 13:21:41 -0700887 *
888 * @param cellX X coordinate of the cell
889 * @param cellY Y coordinate of the cell
890 *
891 * @param result Array of 2 ints to hold the x and y coordinate of the point
892 */
Sebastian Franco45b39b52023-01-10 10:47:46 -0600893 public void regionToCenterPoint(int cellX, int cellY, int spanX, int spanY, int[] result) {
Jon Miranda228877d2021-02-09 11:05:00 -0500894 cellToRect(cellX, cellY, spanX, spanY, mTempRect);
895 result[0] = mTempRect.centerX();
896 result[1] = mTempRect.centerY();
Adam Cohen19f37922012-03-21 11:59:11 -0700897 }
898
Tony Wickham3cfa5ed2021-11-03 13:20:43 -0700899 /**
900 * Returns the distance between the given coordinate and the visual center of the given cell.
901 */
902 public float getDistanceFromWorkspaceCellVisualCenter(float x, float y, int[] cell) {
903 getWorkspaceCellVisualCenter(cell[0], cell[1], mTmpPoint);
Sunny Goyalf7a29e82015-04-24 15:20:43 -0700904 return (float) Math.hypot(x - mTmpPoint[0], y - mTmpPoint[1]);
Adam Cohen482ed822012-03-02 14:15:13 -0800905 }
906
Tony Wickham3cfa5ed2021-11-03 13:20:43 -0700907 private void getWorkspaceCellVisualCenter(int cellX, int cellY, int[] outPoint) {
908 View child = getChildAt(cellX, cellY);
909 if (child instanceof DraggableView) {
910 DraggableView draggableChild = (DraggableView) child;
911 if (draggableChild.getViewType() == DRAGGABLE_ICON) {
912 cellToPoint(cellX, cellY, outPoint);
913 draggableChild.getWorkspaceVisualDragBounds(mTempRect);
914 mTempRect.offset(outPoint[0], outPoint[1]);
915 outPoint[0] = mTempRect.centerX();
916 outPoint[1] = mTempRect.centerY();
917 return;
918 }
919 }
920 cellToCenterPoint(cellX, cellY, outPoint);
921 }
922
Tony Wickham0ac045f2021-11-03 13:17:02 -0700923 /**
924 * Returns the max distance from the center of a cell that can accept a drop to create a folder.
925 */
Tony Wickham12784902021-11-03 14:02:10 -0700926 public float getFolderCreationRadius(int[] targetCell) {
Tony Wickham0ac045f2021-11-03 13:17:02 -0700927 DeviceProfile grid = mActivity.getDeviceProfile();
Tony Wickham12784902021-11-03 14:02:10 -0700928 float iconVisibleRadius = ICON_VISIBLE_AREA_FACTOR * grid.iconSizePx / 2;
929 // Halfway between reorder radius and icon.
Sebastian Franco6e1024e2022-07-29 13:46:49 -0700930 return (getReorderRadius(targetCell, 1, 1) + iconVisibleRadius) / 2;
Tony Wickham12784902021-11-03 14:02:10 -0700931 }
932
933 /**
934 * Returns the max distance from the center of a cell that will start to reorder on drag over.
935 */
Sebastian Franco6e1024e2022-07-29 13:46:49 -0700936 public float getReorderRadius(int[] targetCell, int spanX, int spanY) {
Tony Wickham12784902021-11-03 14:02:10 -0700937 int[] centerPoint = mTmpPoint;
938 getWorkspaceCellVisualCenter(targetCell[0], targetCell[1], centerPoint);
939
940 Rect cellBoundsWithSpacing = mTempRect;
Sebastian Franco6e1024e2022-07-29 13:46:49 -0700941 cellToRect(targetCell[0], targetCell[1], spanX, spanY, cellBoundsWithSpacing);
Tony Wickham12784902021-11-03 14:02:10 -0700942 cellBoundsWithSpacing.inset(-mBorderSpace.x / 2, -mBorderSpace.y / 2);
943
Sebastian Francoc8392ea2022-10-28 16:38:37 -0700944 if (canCreateFolder(getChildAt(targetCell[0], targetCell[1])) && spanX == 1 && spanY == 1) {
Tony Wickham12784902021-11-03 14:02:10 -0700945 // Take only the circle in the smaller dimension, to ensure we don't start reordering
946 // too soon before accepting a folder drop.
947 int minRadius = centerPoint[0] - cellBoundsWithSpacing.left;
948 minRadius = Math.min(minRadius, centerPoint[1] - cellBoundsWithSpacing.top);
949 minRadius = Math.min(minRadius, cellBoundsWithSpacing.right - centerPoint[0]);
950 minRadius = Math.min(minRadius, cellBoundsWithSpacing.bottom - centerPoint[1]);
951 return minRadius;
952 }
953 // Take up the entire cell, including space between this cell and the adjacent ones.
Sebastian Francoc8392ea2022-10-28 16:38:37 -0700954 // Multiply by span to scale radius
955 return (float) Math.hypot(spanX * cellBoundsWithSpacing.width() / 2f,
956 spanY * cellBoundsWithSpacing.height() / 2f);
Tony Wickham0ac045f2021-11-03 13:17:02 -0700957 }
958
Adam Cohenf9c184a2016-01-15 16:47:43 -0800959 public int getCellWidth() {
Romain Guy84f296c2009-11-04 15:00:44 -0800960 return mCellWidth;
961 }
962
Sunny Goyal0b754e52017-08-07 07:42:45 -0700963 public int getCellHeight() {
Romain Guy84f296c2009-11-04 15:00:44 -0800964 return mCellHeight;
965 }
966
Adam Cohenf0f4eda2013-06-06 21:27:03 -0700967 public void setFixedSize(int width, int height) {
968 mFixedWidth = width;
969 mFixedHeight = height;
970 }
971
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800972 @Override
973 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800974 int widthSpecMode = MeasureSpec.getMode(widthMeasureSpec);
The Android Open Source Project31dd5032009-03-03 19:32:27 -0800975 int heightSpecMode = MeasureSpec.getMode(heightMeasureSpec);
Winson Chung5f8afe62013-08-12 16:19:28 -0700976 int widthSize = MeasureSpec.getSize(widthMeasureSpec);
977 int heightSize = MeasureSpec.getSize(heightMeasureSpec);
Winson Chung2d75f122013-09-23 16:53:31 -0700978 int childWidthSize = widthSize - (getPaddingLeft() + getPaddingRight());
979 int childHeightSize = heightSize - (getPaddingTop() + getPaddingBottom());
Sunny Goyalae6e3182019-04-30 12:04:37 -0700980
Winson Chung11a1a532013-09-13 11:14:45 -0700981 if (mFixedCellWidth < 0 || mFixedCellHeight < 0) {
Thales Lima78d00ad2021-09-30 11:29:06 +0100982 int cw = DeviceProfile.calculateCellWidth(childWidthSize, mBorderSpace.x,
Jon Miranda228877d2021-02-09 11:05:00 -0500983 mCountX);
Thales Lima78d00ad2021-09-30 11:29:06 +0100984 int ch = DeviceProfile.calculateCellHeight(childHeightSize, mBorderSpace.y,
Jon Miranda228877d2021-02-09 11:05:00 -0500985 mCountY);
Winson Chung11a1a532013-09-13 11:14:45 -0700986 if (cw != mCellWidth || ch != mCellHeight) {
987 mCellWidth = cw;
988 mCellHeight = ch;
Jon Miranda228877d2021-02-09 11:05:00 -0500989 mShortcutsAndWidgets.setCellDimensions(mCellWidth, mCellHeight, mCountX, mCountY,
Thales Lima78d00ad2021-09-30 11:29:06 +0100990 mBorderSpace);
Winson Chung11a1a532013-09-13 11:14:45 -0700991 }
Winson Chung5f8afe62013-08-12 16:19:28 -0700992 }
Winson Chungaafa03c2010-06-11 17:34:16 -0700993
Winson Chung2d75f122013-09-23 16:53:31 -0700994 int newWidth = childWidthSize;
995 int newHeight = childHeightSize;
Adam Cohenf0f4eda2013-06-06 21:27:03 -0700996 if (mFixedWidth > 0 && mFixedHeight > 0) {
997 newWidth = mFixedWidth;
998 newHeight = mFixedHeight;
999 } else if (widthSpecMode == MeasureSpec.UNSPECIFIED || heightSpecMode == MeasureSpec.UNSPECIFIED) {
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001000 throw new RuntimeException("CellLayout cannot have UNSPECIFIED dimensions");
1001 }
1002
Sunny Goyal4fe5a372015-05-14 19:55:10 -07001003 mShortcutsAndWidgets.measure(
1004 MeasureSpec.makeMeasureSpec(newWidth, MeasureSpec.EXACTLY),
1005 MeasureSpec.makeMeasureSpec(newHeight, MeasureSpec.EXACTLY));
1006
1007 int maxWidth = mShortcutsAndWidgets.getMeasuredWidth();
1008 int maxHeight = mShortcutsAndWidgets.getMeasuredHeight();
Winson Chung2d75f122013-09-23 16:53:31 -07001009 if (mFixedWidth > 0 && mFixedHeight > 0) {
1010 setMeasuredDimension(maxWidth, maxHeight);
1011 } else {
1012 setMeasuredDimension(widthSize, heightSize);
1013 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001014 }
1015
1016 @Override
Michael Jurka28750fb2010-09-24 17:43:49 -07001017 protected void onLayout(boolean changed, int l, int t, int r, int b) {
Tony Wickham26b01422015-11-10 14:44:32 -08001018 int left = getPaddingLeft();
Sunny Goyal7ce471b2017-08-02 03:37:39 -07001019 left += (int) Math.ceil(getUnusedHorizontalSpace() / 2f);
Sunny Goyal7c786f72016-06-01 14:08:21 -07001020 int right = r - l - getPaddingRight();
Sunny Goyal7ce471b2017-08-02 03:37:39 -07001021 right -= (int) Math.ceil(getUnusedHorizontalSpace() / 2f);
Sunny Goyal7c786f72016-06-01 14:08:21 -07001022
Winson Chung38848ca2013-10-08 12:03:44 -07001023 int top = getPaddingTop();
Sunny Goyal7c786f72016-06-01 14:08:21 -07001024 int bottom = b - t - getPaddingBottom();
Sunny Goyal4fe5a372015-05-14 19:55:10 -07001025
Sunny Goyal7c786f72016-06-01 14:08:21 -07001026 // Expand the background drawing bounds by the padding baked into the background drawable
1027 mBackground.getPadding(mTempRect);
1028 mBackground.setBounds(
Jon Miranda28032002017-07-13 16:18:56 -07001029 left - mTempRect.left - getPaddingLeft(),
1030 top - mTempRect.top - getPaddingTop(),
1031 right + mTempRect.right + getPaddingRight(),
1032 bottom + mTempRect.bottom + getPaddingBottom());
Sunny Goyalae6e3182019-04-30 12:04:37 -07001033
Sunny Goyalc4d32012020-04-03 17:10:11 -07001034 mShortcutsAndWidgets.layout(left, top, right, bottom);
Sunny Goyal7c786f72016-06-01 14:08:21 -07001035 }
1036
Tony Wickhama501d492015-11-03 18:05:01 -08001037 /**
1038 * Returns the amount of space left over after subtracting padding and cells. This space will be
1039 * very small, a few pixels at most, and is a result of rounding down when calculating the cell
Jon Miranda228877d2021-02-09 11:05:00 -05001040 * width in {@link DeviceProfile#calculateCellWidth(int, int, int)}.
Tony Wickhama501d492015-11-03 18:05:01 -08001041 */
1042 public int getUnusedHorizontalSpace() {
Jon Miranda228877d2021-02-09 11:05:00 -05001043 return getMeasuredWidth() - getPaddingLeft() - getPaddingRight() - (mCountX * mCellWidth)
Thales Lima78d00ad2021-09-30 11:29:06 +01001044 - ((mCountX - 1) * mBorderSpace.x);
Tony Wickhama501d492015-11-03 18:05:01 -08001045 }
1046
Sunny Goyal2805e632015-05-20 15:35:32 -07001047 @Override
1048 protected boolean verifyDrawable(Drawable who) {
Sunny Goyal7ce471b2017-08-02 03:37:39 -07001049 return super.verifyDrawable(who) || (who == mBackground);
Sunny Goyal2805e632015-05-20 15:35:32 -07001050 }
1051
Michael Jurkaa52570f2012-03-20 03:18:20 -07001052 public ShortcutAndWidgetContainer getShortcutsAndWidgets() {
Sunny Goyaldcbcc862014-08-12 15:58:36 -07001053 return mShortcutsAndWidgets;
Michael Jurkaa52570f2012-03-20 03:18:20 -07001054 }
1055
Jon Miranda228877d2021-02-09 11:05:00 -05001056 public View getChildAt(int cellX, int cellY) {
1057 return mShortcutsAndWidgets.getChildAt(cellX, cellY);
Patrick Dubroy440c3602010-07-13 17:50:32 -07001058 }
1059
Adam Cohen76fc0852011-06-17 13:26:23 -07001060 public boolean animateChildToPosition(final View child, int cellX, int cellY, int duration,
Adam Cohen482ed822012-03-02 14:15:13 -08001061 int delay, boolean permanent, boolean adjustOccupied) {
Michael Jurkaa52570f2012-03-20 03:18:20 -07001062 ShortcutAndWidgetContainer clc = getShortcutsAndWidgets();
Adam Cohen482ed822012-03-02 14:15:13 -08001063
Adam Cohen1d13c0b2020-04-21 16:29:12 -07001064 if (clc.indexOfChild(child) != -1 && (child instanceof Reorderable)) {
Sebastian Francod4682992022-10-05 13:03:09 -05001065 final CellLayoutLayoutParams lp = (CellLayoutLayoutParams) child.getLayoutParams();
Adam Cohenbfbfd262011-06-13 16:55:12 -07001066 final ItemInfo info = (ItemInfo) child.getTag();
Adam Cohen1d13c0b2020-04-21 16:29:12 -07001067 final Reorderable item = (Reorderable) child;
Adam Cohenbfbfd262011-06-13 16:55:12 -07001068
1069 // We cancel any existing animations
1070 if (mReorderAnimators.containsKey(lp)) {
1071 mReorderAnimators.get(lp).cancel();
1072 mReorderAnimators.remove(lp);
1073 }
1074
Adam Cohen1d13c0b2020-04-21 16:29:12 -07001075
Adam Cohen482ed822012-03-02 14:15:13 -08001076 if (adjustOccupied) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001077 GridOccupancy occupied = permanent ? mOccupied : mTmpOccupied;
Sebastian Franco877088e2023-01-03 15:16:22 -07001078 occupied.markCells(lp.getCellX(), lp.getCellY(), lp.cellHSpan, lp.cellVSpan, false);
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001079 occupied.markCells(cellX, cellY, lp.cellHSpan, lp.cellVSpan, true);
Adam Cohen482ed822012-03-02 14:15:13 -08001080 }
Adam Cohen1d13c0b2020-04-21 16:29:12 -07001081
1082 // Compute the new x and y position based on the new cellX and cellY
1083 // We leverage the actual layout logic in the layout params and hence need to modify
1084 // state and revert that state.
1085 final int oldX = lp.x;
1086 final int oldY = lp.y;
Adam Cohenbfbfd262011-06-13 16:55:12 -07001087 lp.isLockedToGrid = true;
Adam Cohen482ed822012-03-02 14:15:13 -08001088 if (permanent) {
Sunny Goyal669b71f2023-01-27 14:37:07 -08001089 lp.setCellX(cellX);
1090 lp.setCellY(cellY);
Adam Cohen482ed822012-03-02 14:15:13 -08001091 } else {
Sebastian Franco877088e2023-01-03 15:16:22 -07001092 lp.setTmpCellX(cellX);
1093 lp.setTmpCellY(cellY);
Adam Cohen482ed822012-03-02 14:15:13 -08001094 }
Jon Mirandae96798e2016-12-07 12:10:44 -08001095 clc.setupLp(child);
Adam Cohen482ed822012-03-02 14:15:13 -08001096 final int newX = lp.x;
1097 final int newY = lp.y;
Adam Cohen76fc0852011-06-17 13:26:23 -07001098 lp.x = oldX;
1099 lp.y = oldY;
Adam Cohen1d13c0b2020-04-21 16:29:12 -07001100 lp.isLockedToGrid = false;
1101 // End compute new x and y
1102
Sunny Goyal82dfc152023-02-24 16:50:09 -08001103 MultiTranslateDelegate mtd = item.getTranslateDelegate();
1104 float initPreviewOffsetX = mtd.getTranslationX(INDEX_REORDER_PREVIEW_OFFSET).getValue();
1105 float initPreviewOffsetY = mtd.getTranslationY(INDEX_REORDER_PREVIEW_OFFSET).getValue();
Adam Cohen1d13c0b2020-04-21 16:29:12 -07001106 final float finalPreviewOffsetX = newX - oldX;
1107 final float finalPreviewOffsetY = newY - oldY;
1108
Adam Cohen482ed822012-03-02 14:15:13 -08001109 // Exit early if we're not actually moving the view
Adam Cohen1d13c0b2020-04-21 16:29:12 -07001110 if (finalPreviewOffsetX == 0 && finalPreviewOffsetY == 0
1111 && initPreviewOffsetX == 0 && initPreviewOffsetY == 0) {
Adam Cohen482ed822012-03-02 14:15:13 -08001112 lp.isLockedToGrid = true;
1113 return true;
1114 }
1115
Sunny Goyal849c6a22018-08-08 16:33:46 -07001116 ValueAnimator va = ValueAnimator.ofFloat(0f, 1f);
Adam Cohen482ed822012-03-02 14:15:13 -08001117 va.setDuration(duration);
1118 mReorderAnimators.put(lp, va);
1119
1120 va.addUpdateListener(new AnimatorUpdateListener() {
1121 @Override
Adam Cohenbfbfd262011-06-13 16:55:12 -07001122 public void onAnimationUpdate(ValueAnimator animation) {
Jon Mirandae96798e2016-12-07 12:10:44 -08001123 float r = (Float) animation.getAnimatedValue();
Adam Cohen1d13c0b2020-04-21 16:29:12 -07001124 float x = (1 - r) * initPreviewOffsetX + r * finalPreviewOffsetX;
1125 float y = (1 - r) * initPreviewOffsetY + r * finalPreviewOffsetY;
Sunny Goyal82dfc152023-02-24 16:50:09 -08001126 item.getTranslateDelegate().setTranslation(INDEX_REORDER_PREVIEW_OFFSET, x, y);
Adam Cohenbfbfd262011-06-13 16:55:12 -07001127 }
1128 });
Adam Cohen482ed822012-03-02 14:15:13 -08001129 va.addListener(new AnimatorListenerAdapter() {
Adam Cohenbfbfd262011-06-13 16:55:12 -07001130 boolean cancelled = false;
1131 public void onAnimationEnd(Animator animation) {
1132 // If the animation was cancelled, it means that another animation
1133 // has interrupted this one, and we don't want to lock the item into
1134 // place just yet.
1135 if (!cancelled) {
1136 lp.isLockedToGrid = true;
Sunny Goyal82dfc152023-02-24 16:50:09 -08001137 item.getTranslateDelegate()
1138 .setTranslation(INDEX_REORDER_PREVIEW_OFFSET, 0, 0);
Adam Cohen482ed822012-03-02 14:15:13 -08001139 child.requestLayout();
Adam Cohenbfbfd262011-06-13 16:55:12 -07001140 }
1141 if (mReorderAnimators.containsKey(lp)) {
1142 mReorderAnimators.remove(lp);
1143 }
1144 }
1145 public void onAnimationCancel(Animator animation) {
1146 cancelled = true;
1147 }
1148 });
Adam Cohen482ed822012-03-02 14:15:13 -08001149 va.setStartDelay(delay);
1150 va.start();
Adam Cohenbfbfd262011-06-13 16:55:12 -07001151 return true;
1152 }
1153 return false;
1154 }
1155
Adam Cohenf7ca3b42021-02-22 11:03:58 -08001156 void visualizeDropLocation(int cellX, int cellY, int spanX, int spanY,
1157 DropTarget.DragObject dragObject) {
1158 if (mDragCell[0] != cellX || mDragCell[1] != cellY || mDragCellSpan[0] != spanX
1159 || mDragCellSpan[1] != spanY) {
Adam Cohen482ed822012-03-02 14:15:13 -08001160 mDragCell[0] = cellX;
1161 mDragCell[1] = cellY;
Adam Cohenf7ca3b42021-02-22 11:03:58 -08001162 mDragCellSpan[0] = spanX;
1163 mDragCellSpan[1] = spanY;
Steven Ng30dd1d62021-03-15 21:45:49 +00001164
Steven Nga999e222021-04-19 18:17:15 +01001165 // Apply color extraction on a widget when dragging.
1166 applyColorExtractionOnWidget(dragObject, mDragCell, spanX, spanY);
1167
Joe Onorato4be866d2010-10-10 11:26:02 -07001168 final int oldIndex = mDragOutlineCurrent;
Patrick Dubroy08ae2ec2010-10-14 23:54:22 -07001169 mDragOutlineAnims[oldIndex].animateOut();
1170 mDragOutlineCurrent = (oldIndex + 1) % mDragOutlines.length;
Sunny Goyal106bf642015-07-16 12:18:06 -07001171
Sebastian Francod4682992022-10-05 13:03:09 -05001172 CellLayoutLayoutParams cell = mDragOutlines[mDragOutlineCurrent];
Sebastian Franco877088e2023-01-03 15:16:22 -07001173 cell.setCellX(cellX);
1174 cell.setCellY(cellY);
Adam Cohenf7ca3b42021-02-22 11:03:58 -08001175 cell.cellHSpan = spanX;
1176 cell.cellVSpan = spanY;
Adam Cohen65086992020-02-19 08:40:49 -08001177
Patrick Dubroy08ae2ec2010-10-14 23:54:22 -07001178 mDragOutlineAnims[mDragOutlineCurrent].animateIn();
Adam Cohenf7ca3b42021-02-22 11:03:58 -08001179 invalidate();
Sunny Goyale78e3d72015-09-24 11:23:31 -07001180
1181 if (dragObject.stateAnnouncer != null) {
Sunny Goyalc13403c2016-11-18 23:44:48 -08001182 dragObject.stateAnnouncer.announce(getItemMoveDescription(cellX, cellY));
Sunny Goyale78e3d72015-09-24 11:23:31 -07001183 }
Adam Cohenf7ca3b42021-02-22 11:03:58 -08001184
Patrick Dubroy6569f2c2010-07-12 14:25:18 -07001185 }
1186 }
1187
Steven Ng30dd1d62021-03-15 21:45:49 +00001188 /** Applies the local color extraction to a dragging widget object. */
Steven Nga999e222021-04-19 18:17:15 +01001189 private void applyColorExtractionOnWidget(DropTarget.DragObject dragObject, int[] targetCell,
1190 int spanX, int spanY) {
Steven Ng30dd1d62021-03-15 21:45:49 +00001191 // Apply local extracted color if the DragView is an AppWidgetHostViewDrawable.
Steven Ng32427202021-04-19 18:12:12 +01001192 View view = dragObject.dragView.getContentView();
1193 if (view instanceof LauncherAppWidgetHostView) {
Andras Kloczl2dacbee2022-02-21 16:53:28 +00001194 int screenId = getWorkspace().getIdForScreen(this);
Steven Ng30dd1d62021-03-15 21:45:49 +00001195 cellToRect(targetCell[0], targetCell[1], spanX, spanY, mTempRect);
Jonathan Miranda21930da2021-05-03 18:44:13 +00001196
Sunny Goyal69a8eec2021-07-22 10:02:50 -07001197 ((LauncherAppWidgetHostView) view).handleDrag(mTempRect, this, screenId);
Steven Ng30dd1d62021-03-15 21:45:49 +00001198 }
1199 }
1200
Sunny Goyal726bee72018-03-05 12:54:24 -08001201 @SuppressLint("StringFormatMatches")
Sunny Goyalc13403c2016-11-18 23:44:48 -08001202 public String getItemMoveDescription(int cellX, int cellY) {
1203 if (mContainerType == HOTSEAT) {
1204 return getContext().getString(R.string.move_to_hotseat_position,
1205 Math.max(cellX, cellY) + 1);
1206 } else {
Shikha Malhotraf78da1b2022-04-11 10:23:18 +00001207 Workspace<?> workspace = getWorkspace();
Andras Kloczl2dacbee2022-02-21 16:53:28 +00001208 int row = cellY + 1;
1209 int col = workspace.mIsRtl ? mCountX - cellX : cellX + 1;
1210 int panelCount = workspace.getPanelCount();
Sebastian Franco930531f2022-06-16 16:49:11 -07001211 int screenId = workspace.getIdForScreen(this);
1212 int pageIndex = workspace.getPageIndexForScreenId(screenId);
Andras Kloczl2dacbee2022-02-21 16:53:28 +00001213 if (panelCount > 1) {
1214 // Increment the column if the target is on the right side of a two panel home
Andras Kloczl2dacbee2022-02-21 16:53:28 +00001215 col += (pageIndex % panelCount) * mCountX;
1216 }
Sebastian Franco930531f2022-06-16 16:49:11 -07001217 return getContext().getString(R.string.move_to_empty_cell_description, row, col,
1218 workspace.getPageDescription(pageIndex));
Sunny Goyalc13403c2016-11-18 23:44:48 -08001219 }
1220 }
1221
Shikha Malhotraf78da1b2022-04-11 10:23:18 +00001222 private Workspace<?> getWorkspace() {
Andras Kloczl2dacbee2022-02-21 16:53:28 +00001223 return Launcher.cast(mActivity).getWorkspace();
1224 }
1225
Adam Cohene0310962011-04-18 16:15:31 -07001226 public void clearDragOutlines() {
1227 final int oldIndex = mDragOutlineCurrent;
1228 mDragOutlineAnims[oldIndex].animateOut();
Adam Cohend41fbf52012-02-16 23:53:59 -08001229 mDragCell[0] = mDragCell[1] = -1;
Adam Cohene0310962011-04-18 16:15:31 -07001230 }
1231
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001232 /**
Jeff Sharkey70864282009-04-07 21:08:40 -07001233 * Find a vacant area that will fit the given bounds nearest the requested
1234 * cell location. Uses Euclidean distance to score multiple vacant areas.
Winson Chungaafa03c2010-06-11 17:34:16 -07001235 *
Romain Guy51afc022009-05-04 18:03:43 -07001236 * @param pixelX The X location at which you want to search for a vacant area.
1237 * @param pixelY The Y location at which you want to search for a vacant area.
Adam Cohend41fbf52012-02-16 23:53:59 -08001238 * @param minSpanX The minimum horizontal span required
1239 * @param minSpanY The minimum vertical span required
1240 * @param spanX Horizontal span of the object.
1241 * @param spanY Vertical span of the object.
1242 * @param result Array in which to place the result, or null (in which case a new array will
1243 * be allocated)
1244 * @return The X, Y cell of a vacant area that can contain this object,
1245 * nearest the requested location.
1246 */
Sebastian Francoe4c03452022-12-27 14:50:02 -06001247 public int[] findNearestVacantArea(int pixelX, int pixelY, int minSpanX, int minSpanY,
1248 int spanX, int spanY, int[] result, int[] resultSpan) {
Sebastian Francob57c0b22022-06-28 13:54:35 -07001249 return findNearestArea(pixelX, pixelY, minSpanX, minSpanY, spanX, spanY, false,
Adam Cohend41fbf52012-02-16 23:53:59 -08001250 result, resultSpan);
1251 }
1252
Adam Cohend41fbf52012-02-16 23:53:59 -08001253 /**
1254 * Find a vacant area that will fit the given bounds nearest the requested
1255 * cell location. Uses Euclidean distance to score multiple vacant areas.
Sebastian Francob57c0b22022-06-28 13:54:35 -07001256 * @param relativeXPos The X location relative to the Cell layout at which you want to search
1257 * for a vacant area.
1258 * @param relativeYPos The Y location relative to the Cell layout at which you want to search
1259 * for a vacant area.
Adam Cohend41fbf52012-02-16 23:53:59 -08001260 * @param minSpanX The minimum horizontal span required
1261 * @param minSpanY The minimum vertical span required
1262 * @param spanX Horizontal span of the object.
1263 * @param spanY Vertical span of the object.
1264 * @param ignoreOccupied If true, the result can be an occupied cell
1265 * @param result Array in which to place the result, or null (in which case a new array will
1266 * be allocated)
1267 * @return The X, Y cell of a vacant area that can contain this object,
1268 * nearest the requested location.
1269 */
Sebastian Franco96c46e72023-05-08 10:04:44 -06001270 protected int[] findNearestArea(int relativeXPos, int relativeYPos, int minSpanX, int minSpanY,
Sebastian Francob57c0b22022-06-28 13:54:35 -07001271 int spanX, int spanY, boolean ignoreOccupied, int[] result, int[] resultSpan) {
Sebastian Francob57c0b22022-06-28 13:54:35 -07001272 // For items with a spanX / spanY > 1, the passed in point (relativeXPos, relativeYPos)
1273 // corresponds to the center of the item, but we are searching based on the top-left cell,
1274 // so we translate the point over to correspond to the top-left.
1275 relativeXPos = (int) (relativeXPos - (mCellWidth + mBorderSpace.x) * (spanX - 1) / 2f);
1276 relativeYPos = (int) (relativeYPos - (mCellHeight + mBorderSpace.y) * (spanY - 1) / 2f);
Adam Cohene3e27a82011-04-15 12:07:39 -07001277
Jeff Sharkey70864282009-04-07 21:08:40 -07001278 // Keep track of best-scoring drop area
Patrick Dubroyde7658b2010-09-27 11:15:43 -07001279 final int[] bestXY = result != null ? result : new int[2];
Jeff Sharkey70864282009-04-07 21:08:40 -07001280 double bestDistance = Double.MAX_VALUE;
Adam Cohend41fbf52012-02-16 23:53:59 -08001281 final Rect bestRect = new Rect(-1, -1, -1, -1);
Rajeev Kumar9962dbe2017-06-12 12:16:20 -07001282 final Stack<Rect> validRegions = new Stack<>();
Winson Chungaafa03c2010-06-11 17:34:16 -07001283
Patrick Dubroyde7658b2010-09-27 11:15:43 -07001284 final int countX = mCountX;
1285 final int countY = mCountY;
Patrick Dubroyde7658b2010-09-27 11:15:43 -07001286
Adam Cohend41fbf52012-02-16 23:53:59 -08001287 if (minSpanX <= 0 || minSpanY <= 0 || spanX <= 0 || spanY <= 0 ||
1288 spanX < minSpanX || spanY < minSpanY) {
1289 return bestXY;
1290 }
1291
1292 for (int y = 0; y < countY - (minSpanY - 1); y++) {
Michael Jurkac28de512010-08-13 11:27:44 -07001293 inner:
Adam Cohend41fbf52012-02-16 23:53:59 -08001294 for (int x = 0; x < countX - (minSpanX - 1); x++) {
1295 int ySize = -1;
1296 int xSize = -1;
Sebastian Francob57c0b22022-06-28 13:54:35 -07001297 if (!ignoreOccupied) {
Adam Cohend41fbf52012-02-16 23:53:59 -08001298 // First, let's see if this thing fits anywhere
1299 for (int i = 0; i < minSpanX; i++) {
1300 for (int j = 0; j < minSpanY; j++) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001301 if (mOccupied.cells[x + i][y + j]) {
Adam Cohendf035382011-04-11 17:22:04 -07001302 continue inner;
1303 }
Michael Jurkac28de512010-08-13 11:27:44 -07001304 }
1305 }
Adam Cohend41fbf52012-02-16 23:53:59 -08001306 xSize = minSpanX;
1307 ySize = minSpanY;
1308
1309 // We know that the item will fit at _some_ acceptable size, now let's see
1310 // how big we can make it. We'll alternate between incrementing x and y spans
1311 // until we hit a limit.
1312 boolean incX = true;
1313 boolean hitMaxX = xSize >= spanX;
1314 boolean hitMaxY = ySize >= spanY;
1315 while (!(hitMaxX && hitMaxY)) {
1316 if (incX && !hitMaxX) {
1317 for (int j = 0; j < ySize; j++) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001318 if (x + xSize > countX -1 || mOccupied.cells[x + xSize][y + j]) {
Adam Cohend41fbf52012-02-16 23:53:59 -08001319 // We can't move out horizontally
1320 hitMaxX = true;
1321 }
1322 }
1323 if (!hitMaxX) {
1324 xSize++;
1325 }
1326 } else if (!hitMaxY) {
1327 for (int i = 0; i < xSize; i++) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001328 if (y + ySize > countY - 1 || mOccupied.cells[x + i][y + ySize]) {
Adam Cohend41fbf52012-02-16 23:53:59 -08001329 // We can't move out vertically
1330 hitMaxY = true;
1331 }
1332 }
1333 if (!hitMaxY) {
1334 ySize++;
1335 }
1336 }
1337 hitMaxX |= xSize >= spanX;
1338 hitMaxY |= ySize >= spanY;
1339 incX = !incX;
1340 }
Michael Jurkac28de512010-08-13 11:27:44 -07001341 }
Sunny Goyal2805e632015-05-20 15:35:32 -07001342 final int[] cellXY = mTmpPoint;
Adam Cohene3e27a82011-04-15 12:07:39 -07001343 cellToCenterPoint(x, y, cellXY);
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001344
Adam Cohend41fbf52012-02-16 23:53:59 -08001345 // We verify that the current rect is not a sub-rect of any of our previous
1346 // candidates. In this case, the current rect is disqualified in favour of the
1347 // containing rect.
Sebastian Franco4a922672022-10-27 16:42:24 -07001348 Rect currentRect = new Rect(x, y, x + xSize, y + ySize);
Adam Cohend41fbf52012-02-16 23:53:59 -08001349 boolean contained = false;
1350 for (Rect r : validRegions) {
1351 if (r.contains(currentRect)) {
1352 contained = true;
1353 break;
1354 }
1355 }
1356 validRegions.push(currentRect);
Sebastian Francob57c0b22022-06-28 13:54:35 -07001357 double distance = Math.hypot(cellXY[0] - relativeXPos, cellXY[1] - relativeYPos);
Adam Cohen482ed822012-03-02 14:15:13 -08001358
Adam Cohend41fbf52012-02-16 23:53:59 -08001359 if ((distance <= bestDistance && !contained) ||
1360 currentRect.contains(bestRect)) {
Michael Jurkac28de512010-08-13 11:27:44 -07001361 bestDistance = distance;
1362 bestXY[0] = x;
1363 bestXY[1] = y;
Adam Cohend41fbf52012-02-16 23:53:59 -08001364 if (resultSpan != null) {
1365 resultSpan[0] = xSize;
1366 resultSpan[1] = ySize;
1367 }
1368 bestRect.set(currentRect);
Michael Jurkac28de512010-08-13 11:27:44 -07001369 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001370 }
1371 }
1372
Adam Cohenc0dcf592011-06-01 15:30:43 -07001373 // Return -1, -1 if no suitable location found
1374 if (bestDistance == Double.MAX_VALUE) {
1375 bestXY[0] = -1;
1376 bestXY[1] = -1;
Jeff Sharkey70864282009-04-07 21:08:40 -07001377 }
Adam Cohenc0dcf592011-06-01 15:30:43 -07001378 return bestXY;
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001379 }
Winson Chungaafa03c2010-06-11 17:34:16 -07001380
Sebastian Francoe4c03452022-12-27 14:50:02 -06001381 public GridOccupancy getOccupied() {
1382 return mOccupied;
1383 }
1384
Adam Cohen482ed822012-03-02 14:15:13 -08001385 private void copySolutionToTempState(ItemConfiguration solution, View dragView) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001386 mTmpOccupied.clear();
Adam Cohen482ed822012-03-02 14:15:13 -08001387
Michael Jurkaa52570f2012-03-20 03:18:20 -07001388 int childCount = mShortcutsAndWidgets.getChildCount();
Adam Cohen482ed822012-03-02 14:15:13 -08001389 for (int i = 0; i < childCount; i++) {
Michael Jurkaa52570f2012-03-20 03:18:20 -07001390 View child = mShortcutsAndWidgets.getChildAt(i);
Adam Cohen482ed822012-03-02 14:15:13 -08001391 if (child == dragView) continue;
Sebastian Francod4682992022-10-05 13:03:09 -05001392 CellLayoutLayoutParams lp = (CellLayoutLayoutParams) child.getLayoutParams();
Adam Cohen8baab352012-03-20 17:39:21 -07001393 CellAndSpan c = solution.map.get(child);
1394 if (c != null) {
Sebastian Franco877088e2023-01-03 15:16:22 -07001395 lp.setTmpCellX(c.cellX);
1396 lp.setTmpCellY(c.cellY);
Adam Cohen8baab352012-03-20 17:39:21 -07001397 lp.cellHSpan = c.spanX;
1398 lp.cellVSpan = c.spanY;
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001399 mTmpOccupied.markCells(c, true);
Adam Cohen482ed822012-03-02 14:15:13 -08001400 }
1401 }
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001402 mTmpOccupied.markCells(solution, true);
Adam Cohen482ed822012-03-02 14:15:13 -08001403 }
1404
1405 private void animateItemsToSolution(ItemConfiguration solution, View dragView, boolean
1406 commitDragView) {
1407
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001408 GridOccupancy occupied = DESTRUCTIVE_REORDER ? mOccupied : mTmpOccupied;
1409 occupied.clear();
Adam Cohen482ed822012-03-02 14:15:13 -08001410
Michael Jurkaa52570f2012-03-20 03:18:20 -07001411 int childCount = mShortcutsAndWidgets.getChildCount();
Adam Cohen482ed822012-03-02 14:15:13 -08001412 for (int i = 0; i < childCount; i++) {
Michael Jurkaa52570f2012-03-20 03:18:20 -07001413 View child = mShortcutsAndWidgets.getChildAt(i);
Adam Cohen482ed822012-03-02 14:15:13 -08001414 if (child == dragView) continue;
Adam Cohen8baab352012-03-20 17:39:21 -07001415 CellAndSpan c = solution.map.get(child);
1416 if (c != null) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001417 animateChildToPosition(child, c.cellX, c.cellY, REORDER_ANIMATION_DURATION, 0,
Adam Cohen19f37922012-03-21 11:59:11 -07001418 DESTRUCTIVE_REORDER, false);
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001419 occupied.markCells(c, true);
Adam Cohen482ed822012-03-02 14:15:13 -08001420 }
1421 }
1422 if (commitDragView) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001423 occupied.markCells(solution, true);
Adam Cohen482ed822012-03-02 14:15:13 -08001424 }
1425 }
1426
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001427
1428 // This method starts or changes the reorder preview animations
1429 private void beginOrAdjustReorderPreviewAnimations(ItemConfiguration solution,
Adam Cohen65086992020-02-19 08:40:49 -08001430 View dragView, int mode) {
Adam Cohen19f37922012-03-21 11:59:11 -07001431 int childCount = mShortcutsAndWidgets.getChildCount();
Adam Cohen19f37922012-03-21 11:59:11 -07001432 for (int i = 0; i < childCount; i++) {
1433 View child = mShortcutsAndWidgets.getChildAt(i);
1434 if (child == dragView) continue;
1435 CellAndSpan c = solution.map.get(child);
Sebastian Franco5f0af4f2023-11-21 10:45:45 -06001436 boolean skip = mode == ReorderPreviewAnimation.MODE_HINT
1437 && !solution.intersectingViews.contains(child);
Adam Cohend9162062020-03-24 16:35:35 -07001438
Sebastian Francod4682992022-10-05 13:03:09 -05001439 CellLayoutLayoutParams lp = (CellLayoutLayoutParams) child.getLayoutParams();
Adam Cohend9162062020-03-24 16:35:35 -07001440 if (c != null && !skip && (child instanceof Reorderable)) {
Sunny Goyal82dfc152023-02-24 16:50:09 -08001441 ReorderPreviewAnimation rha = new ReorderPreviewAnimation(child,
Sebastian Franco877088e2023-01-03 15:16:22 -07001442 mode, lp.getCellX(), lp.getCellY(), c.cellX, c.cellY, c.spanX, c.spanY);
Adam Cohend024f982012-05-23 18:26:45 -07001443 rha.animate();
Adam Cohen19f37922012-03-21 11:59:11 -07001444 }
1445 }
1446 }
1447
Sunny Goyal849c6a22018-08-08 16:33:46 -07001448 private static final Property<ReorderPreviewAnimation, Float> ANIMATION_PROGRESS =
1449 new Property<ReorderPreviewAnimation, Float>(float.class, "animationProgress") {
1450 @Override
1451 public Float get(ReorderPreviewAnimation anim) {
1452 return anim.animationProgress;
1453 }
1454
1455 @Override
1456 public void set(ReorderPreviewAnimation anim, Float progress) {
1457 anim.setAnimationProgress(progress);
1458 }
1459 };
1460
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001461 // Class which represents the reorder preview animations. These animations show that an item is
Adam Cohen19f37922012-03-21 11:59:11 -07001462 // in a temporary state, and hint at where the item will return to.
Sunny Goyal82dfc152023-02-24 16:50:09 -08001463 class ReorderPreviewAnimation<T extends View & Reorderable> {
1464 final T child;
Adam Cohend024f982012-05-23 18:26:45 -07001465 float finalDeltaX;
1466 float finalDeltaY;
1467 float initDeltaX;
1468 float initDeltaY;
Rajeev Kumar9962dbe2017-06-12 12:16:20 -07001469 final float finalScale;
Adam Cohend024f982012-05-23 18:26:45 -07001470 float initScale;
Rajeev Kumar9962dbe2017-06-12 12:16:20 -07001471 final int mode;
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001472 boolean repeating = false;
1473 private static final int PREVIEW_DURATION = 300;
1474 private static final int HINT_DURATION = Workspace.REORDER_TIMEOUT;
1475
Jon Miranda21266912016-12-19 14:12:05 -08001476 private static final float CHILD_DIVIDEND = 4.0f;
1477
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001478 public static final int MODE_HINT = 0;
1479 public static final int MODE_PREVIEW = 1;
1480
Sunny Goyal849c6a22018-08-08 16:33:46 -07001481 float animationProgress = 0;
Sunny Goyalf0b6db72018-08-13 16:10:14 -07001482 ValueAnimator a;
Adam Cohen19f37922012-03-21 11:59:11 -07001483
Sunny Goyal82dfc152023-02-24 16:50:09 -08001484 ReorderPreviewAnimation(View childView, int mode, int cellX0, int cellY0,
Adam Cohend9162062020-03-24 16:35:35 -07001485 int cellX1, int cellY1, int spanX, int spanY) {
Adam Cohen19f37922012-03-21 11:59:11 -07001486 regionToCenterPoint(cellX0, cellY0, spanX, spanY, mTmpPoint);
1487 final int x0 = mTmpPoint[0];
1488 final int y0 = mTmpPoint[1];
1489 regionToCenterPoint(cellX1, cellY1, spanX, spanY, mTmpPoint);
1490 final int x1 = mTmpPoint[0];
1491 final int y1 = mTmpPoint[1];
1492 final int dX = x1 - x0;
1493 final int dY = y1 - y0;
Jon Miranda21266912016-12-19 14:12:05 -08001494
Sunny Goyal82dfc152023-02-24 16:50:09 -08001495 this.child = (T) childView;
Jon Miranda21266912016-12-19 14:12:05 -08001496 this.mode = mode;
Adam Cohend9162062020-03-24 16:35:35 -07001497 finalDeltaX = 0;
1498 finalDeltaY = 0;
Adam Cohen65086992020-02-19 08:40:49 -08001499
Sunny Goyal82dfc152023-02-24 16:50:09 -08001500 MultiTranslateDelegate mtd = child.getTranslateDelegate();
1501 initDeltaX = mtd.getTranslationX(INDEX_REORDER_BOUNCE_OFFSET).getValue();
1502 initDeltaY = mtd.getTranslationY(INDEX_REORDER_BOUNCE_OFFSET).getValue();
Adam Cohen1d13c0b2020-04-21 16:29:12 -07001503 initScale = child.getReorderBounceScale();
Sunny Goyal82dfc152023-02-24 16:50:09 -08001504 finalScale = mChildScale - (CHILD_DIVIDEND / child.getWidth()) * initScale;
Adam Cohend9162062020-03-24 16:35:35 -07001505
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001506 int dir = mode == MODE_HINT ? -1 : 1;
Adam Cohen19f37922012-03-21 11:59:11 -07001507 if (dX == dY && dX == 0) {
1508 } else {
1509 if (dY == 0) {
Adam Cohend9162062020-03-24 16:35:35 -07001510 finalDeltaX = -dir * Math.signum(dX) * mReorderPreviewAnimationMagnitude;
Adam Cohen19f37922012-03-21 11:59:11 -07001511 } else if (dX == 0) {
Adam Cohend9162062020-03-24 16:35:35 -07001512 finalDeltaY = -dir * Math.signum(dY) * mReorderPreviewAnimationMagnitude;
Adam Cohen19f37922012-03-21 11:59:11 -07001513 } else {
1514 double angle = Math.atan( (float) (dY) / dX);
Adam Cohend9162062020-03-24 16:35:35 -07001515 finalDeltaX = (int) (-dir * Math.signum(dX)
1516 * Math.abs(Math.cos(angle) * mReorderPreviewAnimationMagnitude));
1517 finalDeltaY = (int) (-dir * Math.signum(dY)
1518 * Math.abs(Math.sin(angle) * mReorderPreviewAnimationMagnitude));
Adam Cohen19f37922012-03-21 11:59:11 -07001519 }
1520 }
Jon Miranda21266912016-12-19 14:12:05 -08001521 }
1522
Adam Cohend9162062020-03-24 16:35:35 -07001523 void setInitialAnimationValuesToBaseline() {
1524 initScale = mChildScale;
1525 initDeltaX = 0;
1526 initDeltaY = 0;
Adam Cohen19f37922012-03-21 11:59:11 -07001527 }
1528
Adam Cohend024f982012-05-23 18:26:45 -07001529 void animate() {
Adam Cohend9162062020-03-24 16:35:35 -07001530 boolean noMovement = (finalDeltaX == 0) && (finalDeltaY == 0);
Jon Miranda21266912016-12-19 14:12:05 -08001531
Adam Cohen19f37922012-03-21 11:59:11 -07001532 if (mShakeAnimators.containsKey(child)) {
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001533 ReorderPreviewAnimation oldAnimation = mShakeAnimators.get(child);
Adam Cohen19f37922012-03-21 11:59:11 -07001534 mShakeAnimators.remove(child);
Adam Cohend9162062020-03-24 16:35:35 -07001535
Jon Miranda21266912016-12-19 14:12:05 -08001536 if (noMovement) {
Adam Cohend9162062020-03-24 16:35:35 -07001537 // A previous animation for this item exists, and no new animation will exist.
1538 // Finish the old animation smoothly.
1539 oldAnimation.finishAnimation();
Adam Cohene7587d22012-05-24 18:50:02 -07001540 return;
Adam Cohend9162062020-03-24 16:35:35 -07001541 } else {
1542 // A previous animation for this item exists, and a new one will exist. Stop
1543 // the old animation in its tracks, and proceed with the new one.
1544 oldAnimation.cancel();
Adam Cohene7587d22012-05-24 18:50:02 -07001545 }
Adam Cohen19f37922012-03-21 11:59:11 -07001546 }
Jon Miranda21266912016-12-19 14:12:05 -08001547 if (noMovement) {
Adam Cohen19f37922012-03-21 11:59:11 -07001548 return;
1549 }
Adam Cohend9162062020-03-24 16:35:35 -07001550
Sunny Goyal849c6a22018-08-08 16:33:46 -07001551 ValueAnimator va = ObjectAnimator.ofFloat(this, ANIMATION_PROGRESS, 0, 1);
Adam Cohene7587d22012-05-24 18:50:02 -07001552 a = va;
Tony Wickham9e0702f2015-09-02 14:45:39 -07001553
1554 // Animations are disabled in power save mode, causing the repeated animation to jump
1555 // spastically between beginning and end states. Since this looks bad, we don't repeat
1556 // the animation in power save mode.
Sunny Goyaleaf7a952020-07-29 16:54:20 -07001557 if (areAnimatorsEnabled()) {
Tony Wickham9e0702f2015-09-02 14:45:39 -07001558 va.setRepeatMode(ValueAnimator.REVERSE);
1559 va.setRepeatCount(ValueAnimator.INFINITE);
1560 }
1561
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001562 va.setDuration(mode == MODE_HINT ? HINT_DURATION : PREVIEW_DURATION);
Adam Cohend024f982012-05-23 18:26:45 -07001563 va.setStartDelay((int) (Math.random() * 60));
Adam Cohen19f37922012-03-21 11:59:11 -07001564 va.addListener(new AnimatorListenerAdapter() {
1565 public void onAnimationRepeat(Animator animation) {
Adam Cohen19f37922012-03-21 11:59:11 -07001566 // We make sure to end only after a full period
Adam Cohend9162062020-03-24 16:35:35 -07001567 setInitialAnimationValuesToBaseline();
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001568 repeating = true;
Adam Cohen19f37922012-03-21 11:59:11 -07001569 }
1570 });
Adam Cohen19f37922012-03-21 11:59:11 -07001571 mShakeAnimators.put(child, this);
1572 va.start();
1573 }
1574
Sunny Goyal849c6a22018-08-08 16:33:46 -07001575 private void setAnimationProgress(float progress) {
1576 animationProgress = progress;
1577 float r1 = (mode == MODE_HINT && repeating) ? 1.0f : animationProgress;
1578 float x = r1 * finalDeltaX + (1 - r1) * initDeltaX;
1579 float y = r1 * finalDeltaY + (1 - r1) * initDeltaY;
Sunny Goyal82dfc152023-02-24 16:50:09 -08001580 child.getTranslateDelegate().setTranslation(INDEX_REORDER_BOUNCE_OFFSET, x, y);
Sunny Goyal849c6a22018-08-08 16:33:46 -07001581 float s = animationProgress * finalScale + (1 - animationProgress) * initScale;
Adam Cohen1d13c0b2020-04-21 16:29:12 -07001582 child.setReorderBounceScale(s);
Sunny Goyal849c6a22018-08-08 16:33:46 -07001583 }
1584
Adam Cohend024f982012-05-23 18:26:45 -07001585 private void cancel() {
Adam Cohene7587d22012-05-24 18:50:02 -07001586 if (a != null) {
1587 a.cancel();
1588 }
Adam Cohen19f37922012-03-21 11:59:11 -07001589 }
Adam Cohene7587d22012-05-24 18:50:02 -07001590
Adam Cohend9162062020-03-24 16:35:35 -07001591 /**
1592 * Smoothly returns the item to its baseline position / scale
1593 */
1594 @Thunk void finishAnimation() {
Adam Cohene7587d22012-05-24 18:50:02 -07001595 if (a != null) {
1596 a.cancel();
1597 }
Brandon Keely50e6e562012-05-08 16:28:49 -07001598
Adam Cohend9162062020-03-24 16:35:35 -07001599 setInitialAnimationValuesToBaseline();
1600 ValueAnimator va = ObjectAnimator.ofFloat(this, ANIMATION_PROGRESS,
1601 animationProgress, 0);
1602 a = va;
Kateryna Ivanova71203732023-05-24 15:09:00 +00001603 a.setInterpolator(DECELERATE_1_5);
Adam Cohend9162062020-03-24 16:35:35 -07001604 a.setDuration(REORDER_ANIMATION_DURATION);
Sunny Goyal5d2fc322015-07-06 22:52:49 -07001605 a.start();
Brandon Keely50e6e562012-05-08 16:28:49 -07001606 }
Adam Cohen19f37922012-03-21 11:59:11 -07001607 }
1608
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001609 private void completeAndClearReorderPreviewAnimations() {
1610 for (ReorderPreviewAnimation a: mShakeAnimators.values()) {
Adam Cohend9162062020-03-24 16:35:35 -07001611 a.finishAnimation();
Adam Cohen19f37922012-03-21 11:59:11 -07001612 }
1613 mShakeAnimators.clear();
1614 }
1615
Sunny Goyal711c5962021-06-23 12:36:18 -07001616 private void commitTempPlacement(View dragView) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001617 mTmpOccupied.copyTo(mOccupied);
Sunny Goyalaa8ef112015-06-12 20:04:41 -07001618
Andras Kloczl2dacbee2022-02-21 16:53:28 +00001619 int screenId = getWorkspace().getIdForScreen(this);
Sunny Goyalaa8ef112015-06-12 20:04:41 -07001620 int container = Favorites.CONTAINER_DESKTOP;
1621
Sunny Goyalc13403c2016-11-18 23:44:48 -08001622 if (mContainerType == HOTSEAT) {
Sunny Goyalaa8ef112015-06-12 20:04:41 -07001623 screenId = -1;
1624 container = Favorites.CONTAINER_HOTSEAT;
1625 }
1626
Michael Jurkaa52570f2012-03-20 03:18:20 -07001627 int childCount = mShortcutsAndWidgets.getChildCount();
Adam Cohen482ed822012-03-02 14:15:13 -08001628 for (int i = 0; i < childCount; i++) {
Adam Cohenea889a22012-03-27 16:45:39 -07001629 View child = mShortcutsAndWidgets.getChildAt(i);
Sebastian Francod4682992022-10-05 13:03:09 -05001630 CellLayoutLayoutParams lp = (CellLayoutLayoutParams) child.getLayoutParams();
Adam Cohenea889a22012-03-27 16:45:39 -07001631 ItemInfo info = (ItemInfo) child.getTag();
Adam Cohen2acce882012-03-28 19:03:19 -07001632 // We do a null check here because the item info can be null in the case of the
1633 // AllApps button in the hotseat.
Sunny Goyal711c5962021-06-23 12:36:18 -07001634 if (info != null && child != dragView) {
Sunny Goyal669b71f2023-01-27 14:37:07 -08001635 CellPos presenterPos = mActivity.getCellPosMapper().mapModelToPresenter(info);
1636 final boolean requiresDbUpdate = (presenterPos.cellX != lp.getTmpCellX()
1637 || presenterPos.cellY != lp.getTmpCellY() || info.spanX != lp.cellHSpan
1638 || info.spanY != lp.cellVSpan || presenterPos.screenId != screenId);
Sunny Goyalaa8ef112015-06-12 20:04:41 -07001639
Sebastian Franco877088e2023-01-03 15:16:22 -07001640 lp.setCellX(lp.getTmpCellX());
Sebastian Franco877088e2023-01-03 15:16:22 -07001641 lp.setCellY(lp.getTmpCellY());
Sunny Goyalaa8ef112015-06-12 20:04:41 -07001642 if (requiresDbUpdate) {
Sunny Goyalab770a12018-11-14 15:17:26 -08001643 Launcher.cast(mActivity).getModelWriter().modifyItemInDatabase(info, container,
Sunny Goyal669b71f2023-01-27 14:37:07 -08001644 screenId, lp.getCellX(), lp.getCellY(), lp.cellHSpan, lp.cellVSpan);
Sunny Goyalaa8ef112015-06-12 20:04:41 -07001645 }
Adam Cohen2acce882012-03-28 19:03:19 -07001646 }
Adam Cohen482ed822012-03-02 14:15:13 -08001647 }
1648 }
1649
Sunny Goyalf7a29e82015-04-24 15:20:43 -07001650 private void setUseTempCoords(boolean useTempCoords) {
Michael Jurkaa52570f2012-03-20 03:18:20 -07001651 int childCount = mShortcutsAndWidgets.getChildCount();
Adam Cohen482ed822012-03-02 14:15:13 -08001652 for (int i = 0; i < childCount; i++) {
Sebastian Francod4682992022-10-05 13:03:09 -05001653 CellLayoutLayoutParams lp = (CellLayoutLayoutParams) mShortcutsAndWidgets.getChildAt(
1654 i).getLayoutParams();
Adam Cohen482ed822012-03-02 14:15:13 -08001655 lp.useTmpCoords = useTempCoords;
1656 }
1657 }
1658
Sebastian Franco5c8f8682023-11-14 09:52:41 -06001659 /**
1660 * For a given region, return the rectangle of the overlapping cell and span with the given
1661 * region including the region itself. If there is no overlap the rectangle will be
1662 * invalid i.e. -1, 0, -1, 0.
1663 */
1664 @Nullable
1665 public Rect getIntersectingRectanglesInRegion(final Rect region, final View dragView) {
1666 Rect boundingRect = new Rect(region);
Adam Cohen19f37922012-03-21 11:59:11 -07001667 Rect r1 = new Rect();
Sebastian Franco5c8f8682023-11-14 09:52:41 -06001668 boolean isOverlapping = false;
Adam Cohen19f37922012-03-21 11:59:11 -07001669 final int count = mShortcutsAndWidgets.getChildCount();
1670 for (int i = 0; i < count; i++) {
1671 View child = mShortcutsAndWidgets.getChildAt(i);
1672 if (child == dragView) continue;
Sebastian Francod4682992022-10-05 13:03:09 -05001673 CellLayoutLayoutParams
1674 lp = (CellLayoutLayoutParams) child.getLayoutParams();
Sebastian Franco877088e2023-01-03 15:16:22 -07001675 r1.set(lp.getCellX(), lp.getCellY(), lp.getCellX() + lp.cellHSpan,
1676 lp.getCellY() + lp.cellVSpan);
Sebastian Franco5c8f8682023-11-14 09:52:41 -06001677 if (Rect.intersects(region, r1)) {
1678 isOverlapping = true;
1679 boundingRect.union(r1);
Adam Cohen19f37922012-03-21 11:59:11 -07001680 }
1681 }
Sebastian Franco5c8f8682023-11-14 09:52:41 -06001682 return isOverlapping ? boundingRect : null;
Adam Cohen19f37922012-03-21 11:59:11 -07001683 }
1684
Sebastian Francoe4c03452022-12-27 14:50:02 -06001685 public boolean isNearestDropLocationOccupied(int pixelX, int pixelY, int spanX, int spanY,
Adam Cohen19f37922012-03-21 11:59:11 -07001686 View dragView, int[] result) {
Sebastián Francof9a6ac22022-11-15 22:56:37 +00001687 result = findNearestAreaIgnoreOccupied(pixelX, pixelY, spanX, spanY, result);
Sebastian Franco5c8f8682023-11-14 09:52:41 -06001688 return getIntersectingRectanglesInRegion(
1689 new Rect(result[0], result[1], result[0] + spanX, result[1] + spanY),
1690 dragView
1691 ) != null;
Adam Cohen19f37922012-03-21 11:59:11 -07001692 }
1693
1694 void revertTempState() {
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001695 completeAndClearReorderPreviewAnimations();
1696 if (isItemPlacementDirty() && !DESTRUCTIVE_REORDER) {
1697 final int count = mShortcutsAndWidgets.getChildCount();
1698 for (int i = 0; i < count; i++) {
1699 View child = mShortcutsAndWidgets.getChildAt(i);
Sebastian Francod4682992022-10-05 13:03:09 -05001700 CellLayoutLayoutParams
1701 lp = (CellLayoutLayoutParams) child.getLayoutParams();
Sebastian Franco877088e2023-01-03 15:16:22 -07001702 if (lp.getTmpCellX() != lp.getCellX() || lp.getTmpCellY() != lp.getCellY()) {
1703 lp.setTmpCellX(lp.getCellX());
1704 lp.setTmpCellY(lp.getCellY());
1705 animateChildToPosition(child, lp.getCellX(), lp.getCellY(),
1706 REORDER_ANIMATION_DURATION, 0, false, false);
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001707 }
Adam Cohen19f37922012-03-21 11:59:11 -07001708 }
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001709 setItemPlacementDirty(false);
Adam Cohen19f37922012-03-21 11:59:11 -07001710 }
Adam Cohen19f37922012-03-21 11:59:11 -07001711 }
1712
Adam Cohenbebf0422012-04-11 18:06:28 -07001713 boolean createAreaForResize(int cellX, int cellY, int spanX, int spanY,
1714 View dragView, int[] direction, boolean commit) {
1715 int[] pixelXY = new int[2];
1716 regionToCenterPoint(cellX, cellY, spanX, spanY, pixelXY);
1717
1718 // First we determine if things have moved enough to cause a different layout
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001719 ItemConfiguration swapSolution = findReorderSolution(pixelXY[0], pixelXY[1], spanX, spanY,
Sebastian Francof153d912022-04-22 16:15:27 -05001720 spanX, spanY, direction, dragView, true, new ItemConfiguration());
Adam Cohenbebf0422012-04-11 18:06:28 -07001721
1722 setUseTempCoords(true);
1723 if (swapSolution != null && swapSolution.isSolution) {
1724 // If we're just testing for a possible location (MODE_ACCEPT_DROP), we don't bother
1725 // committing anything or animating anything as we just want to determine if a solution
1726 // exists
1727 copySolutionToTempState(swapSolution, dragView);
1728 setItemPlacementDirty(true);
1729 animateItemsToSolution(swapSolution, dragView, commit);
1730
1731 if (commit) {
Sunny Goyal711c5962021-06-23 12:36:18 -07001732 commitTempPlacement(null);
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001733 completeAndClearReorderPreviewAnimations();
Adam Cohenbebf0422012-04-11 18:06:28 -07001734 setItemPlacementDirty(false);
1735 } else {
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001736 beginOrAdjustReorderPreviewAnimations(swapSolution, dragView,
Adam Cohen65086992020-02-19 08:40:49 -08001737 ReorderPreviewAnimation.MODE_PREVIEW);
Adam Cohenbebf0422012-04-11 18:06:28 -07001738 }
1739 mShortcutsAndWidgets.requestLayout();
1740 }
1741 return swapSolution.isSolution;
1742 }
1743
Sebastian Francoe4c03452022-12-27 14:50:02 -06001744 public ReorderAlgorithm createReorderAlgorithm() {
1745 return new ReorderAlgorithm(this);
1746 }
1747
Sebastian Franco09589322022-11-02 15:25:58 -07001748 protected ItemConfiguration findReorderSolution(int pixelX, int pixelY, int minSpanX,
1749 int minSpanY, int spanX, int spanY, int[] direction, View dragView, boolean decX,
1750 ItemConfiguration solution) {
Sebastian Francoe4c03452022-12-27 14:50:02 -06001751 return createReorderAlgorithm().findReorderSolution(pixelX, pixelY, minSpanX, minSpanY,
1752 spanX, spanY, direction, dragView, decX, solution);
Sebastian Franco09589322022-11-02 15:25:58 -07001753 }
1754
Sebastián Franco61fbd182023-11-29 21:30:43 +00001755 public void copyCurrentStateToSolution(ItemConfiguration solution) {
Sebastian Franco9cab1c32022-10-25 17:28:54 -07001756 int childCount = mShortcutsAndWidgets.getChildCount();
1757 for (int i = 0; i < childCount; i++) {
1758 View child = mShortcutsAndWidgets.getChildAt(i);
1759 CellLayoutLayoutParams lp = (CellLayoutLayoutParams) child.getLayoutParams();
Sebastián Franco61fbd182023-11-29 21:30:43 +00001760 solution.add(child,
1761 new CellAndSpan(lp.getCellX(), lp.getCellY(), lp.cellHSpan, lp.cellVSpan));
Sebastian Franco9cab1c32022-10-25 17:28:54 -07001762 }
Sebastian Franco53a15a42022-10-25 17:28:54 -07001763 }
1764
1765 /**
Sebastian Franco53a15a42022-10-25 17:28:54 -07001766 * When the user drags an Item in the workspace sometimes we need to move the items already in
1767 * the workspace to make space for the new item, this function return a solution for that
1768 * reorder.
1769 *
1770 * @param pixelX X coordinate in the screen of the dragView in pixels
1771 * @param pixelY Y coordinate in the screen of the dragView in pixels
1772 * @param minSpanX minimum horizontal span the item can be shrunk to
1773 * @param minSpanY minimum vertical span the item can be shrunk to
1774 * @param spanX occupied horizontal span
1775 * @param spanY occupied vertical span
1776 * @param dragView the view of the item being draged
1777 * @return returns a solution for the given parameters, the solution contains all the icons and
1778 * the locations they should be in the given solution.
1779 */
1780 public ItemConfiguration calculateReorder(int pixelX, int pixelY, int minSpanX, int minSpanY,
1781 int spanX, int spanY, View dragView) {
Sebastian Francoe4c03452022-12-27 14:50:02 -06001782 return createReorderAlgorithm().calculateReorder(pixelX, pixelY, minSpanX, minSpanY,
1783 spanX, spanY, dragView);
Sebastian Franco53a15a42022-10-25 17:28:54 -07001784 }
Adam Cohen482ed822012-03-02 14:15:13 -08001785
Sebastian Franco9cab1c32022-10-25 17:28:54 -07001786 int[] performReorder(int pixelX, int pixelY, int minSpanX, int minSpanY, int spanX, int spanY,
1787 View dragView, int[] result, int[] resultSpan, int mode) {
1788 if (resultSpan == null) {
1789 resultSpan = new int[]{-1, -1};
1790 }
1791 if (result == null) {
1792 result = new int[]{-1, -1};
1793 }
Sebastian Franco5d990ee2022-11-01 16:08:24 -07001794
1795 ItemConfiguration finalSolution = null;
1796 // We want the solution to match the animation of the preview and to match the drop so we
1797 // only recalculate in mode MODE_SHOW_REORDER_HINT because that the first one to run in the
1798 // reorder cycle.
1799 if (mode == MODE_SHOW_REORDER_HINT || mPreviousSolution == null) {
1800 finalSolution = calculateReorder(pixelX, pixelY, minSpanX, minSpanY, spanX, spanY,
1801 dragView);
1802 mPreviousSolution = finalSolution;
1803 } else {
Sebastian Franco9cab1c32022-10-25 17:28:54 -07001804 finalSolution = mPreviousSolution;
1805 // We reset this vector after drop
1806 if (mode == MODE_ON_DROP || mode == MODE_ON_DROP_EXTERNAL) {
1807 mPreviousSolution = null;
1808 }
Sebastian Franco9cab1c32022-10-25 17:28:54 -07001809 }
1810
1811 if (finalSolution == null || !finalSolution.isSolution) {
1812 result[0] = result[1] = resultSpan[0] = resultSpan[1] = -1;
1813 } else {
1814 result[0] = finalSolution.cellX;
1815 result[1] = finalSolution.cellY;
1816 resultSpan[0] = finalSolution.spanX;
1817 resultSpan[1] = finalSolution.spanY;
Sebastian Franco9c743272022-11-15 15:03:25 -08001818 performReorder(finalSolution, dragView, mode);
Sebastian Franco9cab1c32022-10-25 17:28:54 -07001819 }
Sebastian Franco9cab1c32022-10-25 17:28:54 -07001820 return result;
1821 }
1822
Sebastian Franco53a15a42022-10-25 17:28:54 -07001823 /**
1824 * Animates and submits in the DB the given ItemConfiguration depending of the mode.
1825 *
1826 * @param solution represents widgets on the screen which the Workspace will animate to and
1827 * would be submitted to the database.
1828 * @param dragView view which is being dragged over the workspace that trigger the reorder
1829 * @param mode depending on the mode different animations would be played and depending on the
1830 * mode the solution would be submitted or not the database.
1831 * The possible modes are {@link MODE_SHOW_REORDER_HINT}, {@link MODE_DRAG_OVER},
1832 * {@link MODE_ON_DROP}, {@link MODE_ON_DROP_EXTERNAL}, {@link MODE_ACCEPT_DROP}
1833 * defined in {@link CellLayout}.
1834 */
Sebastian Francoe4c03452022-12-27 14:50:02 -06001835 public void performReorder(ItemConfiguration solution, View dragView, int mode) {
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001836 if (mode == MODE_SHOW_REORDER_HINT) {
Sebastian Franco53a15a42022-10-25 17:28:54 -07001837 beginOrAdjustReorderPreviewAnimations(solution, dragView,
1838 ReorderPreviewAnimation.MODE_HINT);
1839 return;
1840 }
1841 // If we're just testing for a possible location (MODE_ACCEPT_DROP), we don't bother
1842 // committing anything or animating anything as we just want to determine if a solution
1843 // exists
1844 if (mode == MODE_DRAG_OVER || mode == MODE_ON_DROP || mode == MODE_ON_DROP_EXTERNAL) {
1845 if (!DESTRUCTIVE_REORDER) {
1846 setUseTempCoords(true);
1847 }
1848
1849 if (!DESTRUCTIVE_REORDER) {
1850 copySolutionToTempState(solution, dragView);
1851 }
1852 setItemPlacementDirty(true);
1853 animateItemsToSolution(solution, dragView, mode == MODE_ON_DROP);
1854
1855 if (!DESTRUCTIVE_REORDER
1856 && (mode == MODE_ON_DROP || mode == MODE_ON_DROP_EXTERNAL)) {
1857 // Since the temp solution didn't update dragView, don't commit it either
1858 commitTempPlacement(dragView);
1859 completeAndClearReorderPreviewAnimations();
1860 setItemPlacementDirty(false);
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001861 } else {
Sebastian Franco53a15a42022-10-25 17:28:54 -07001862 beginOrAdjustReorderPreviewAnimations(solution, dragView,
1863 ReorderPreviewAnimation.MODE_PREVIEW);
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001864 }
Adam Cohenfa3c58f2013-12-06 16:10:55 -08001865 }
1866
Sebastian Franco53a15a42022-10-25 17:28:54 -07001867 if (mode == MODE_ON_DROP && !DESTRUCTIVE_REORDER) {
Adam Cohen482ed822012-03-02 14:15:13 -08001868 setUseTempCoords(false);
1869 }
Adam Cohen482ed822012-03-02 14:15:13 -08001870
Michael Jurkaa52570f2012-03-20 03:18:20 -07001871 mShortcutsAndWidgets.requestLayout();
Adam Cohen482ed822012-03-02 14:15:13 -08001872 }
1873
Adam Cohen19f37922012-03-21 11:59:11 -07001874 void setItemPlacementDirty(boolean dirty) {
1875 mItemPlacementDirty = dirty;
Adam Cohen482ed822012-03-02 14:15:13 -08001876 }
Adam Cohen19f37922012-03-21 11:59:11 -07001877 boolean isItemPlacementDirty() {
1878 return mItemPlacementDirty;
Adam Cohen482ed822012-03-02 14:15:13 -08001879 }
1880
Sebastian Francoe4c03452022-12-27 14:50:02 -06001881 /**
Adam Cohendf035382011-04-11 17:22:04 -07001882 * Find a starting cell position that will fit the given bounds nearest the requested
1883 * cell location. Uses Euclidean distance to score multiple vacant areas.
1884 *
1885 * @param pixelX The X location at which you want to search for a vacant area.
1886 * @param pixelY The Y location at which you want to search for a vacant area.
1887 * @param spanX Horizontal span of the object.
1888 * @param spanY Vertical span of the object.
Adam Cohendf035382011-04-11 17:22:04 -07001889 * @param result Previously returned value to possibly recycle.
1890 * @return The X, Y cell of a vacant area that can contain this object,
1891 * nearest the requested location.
1892 */
Sebastián Francof9a6ac22022-11-15 22:56:37 +00001893 public int[] findNearestAreaIgnoreOccupied(int pixelX, int pixelY, int spanX, int spanY,
1894 int[] result) {
Sebastian Francob57c0b22022-06-28 13:54:35 -07001895 return findNearestArea(pixelX, pixelY, spanX, spanY, spanX, spanY, true, result, null);
Adam Cohendf035382011-04-11 17:22:04 -07001896 }
1897
Michael Jurka0280c3b2010-09-17 15:00:07 -07001898 boolean existsEmptyCell() {
1899 return findCellForSpan(null, 1, 1);
1900 }
1901
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001902 /**
Michael Jurka0280c3b2010-09-17 15:00:07 -07001903 * Finds the upper-left coordinate of the first rectangle in the grid that can
1904 * hold a cell of the specified dimensions. If intersectX and intersectY are not -1,
1905 * then this method will only return coordinates for rectangles that contain the cell
1906 * (intersectX, intersectY)
1907 *
1908 * @param cellXY The array that will contain the position of a vacant cell if such a cell
1909 * can be found.
1910 * @param spanX The horizontal span of the cell we want to find.
1911 * @param spanY The vertical span of the cell we want to find.
1912 *
1913 * @return True if a vacant cell of the specified dimension was found, false otherwise.
Patrick Dubroy6569f2c2010-07-12 14:25:18 -07001914 */
Hyunyoung Song3f471442015-04-08 19:01:34 -07001915 public boolean findCellForSpan(int[] cellXY, int spanX, int spanY) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001916 if (cellXY == null) {
1917 cellXY = new int[2];
Michael Jurka0280c3b2010-09-17 15:00:07 -07001918 }
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07001919 return mOccupied.findVacantCell(cellXY, spanX, spanY);
Michael Jurka0280c3b2010-09-17 15:00:07 -07001920 }
1921
1922 /**
Winson Chungc07918d2011-07-01 15:35:26 -07001923 * A drag event has begun over this layout.
1924 * It may have begun over this layout (in which case onDragChild is called first),
1925 * or it may have begun on another layout.
1926 */
1927 void onDragEnter() {
Winson Chungc07918d2011-07-01 15:35:26 -07001928 mDragging = true;
Sebastian Franco5aa71ce2022-12-14 12:13:19 -06001929 mPreviousSolution = null;
Winson Chungc07918d2011-07-01 15:35:26 -07001930 }
1931
1932 /**
Michael Jurka0280c3b2010-09-17 15:00:07 -07001933 * Called when drag has left this CellLayout or has been completed (successfully or not)
1934 */
1935 void onDragExit() {
Joe Onorato4be866d2010-10-10 11:26:02 -07001936 // This can actually be called when we aren't in a drag, e.g. when adding a new
1937 // item to this layout via the customize drawer.
1938 // Guard against that case.
1939 if (mDragging) {
1940 mDragging = false;
Patrick Dubroyde7658b2010-09-27 11:15:43 -07001941 }
Patrick Dubroy08ae2ec2010-10-14 23:54:22 -07001942
1943 // Invalidate the drag data
Sebastian Franco5aa71ce2022-12-14 12:13:19 -06001944 mPreviousSolution = null;
Adam Cohend41fbf52012-02-16 23:53:59 -08001945 mDragCell[0] = mDragCell[1] = -1;
Adam Cohenf7ca3b42021-02-22 11:03:58 -08001946 mDragCellSpan[0] = mDragCellSpan[1] = -1;
Patrick Dubroy08ae2ec2010-10-14 23:54:22 -07001947 mDragOutlineAnims[mDragOutlineCurrent].animateOut();
1948 mDragOutlineCurrent = (mDragOutlineCurrent + 1) % mDragOutlineAnims.length;
Adam Cohen19f37922012-03-21 11:59:11 -07001949 revertTempState();
Michael Jurka33945b22010-12-21 18:19:38 -08001950 setIsDragOverlapping(false);
Patrick Dubroy6569f2c2010-07-12 14:25:18 -07001951 }
1952
1953 /**
Winson Chungaafa03c2010-06-11 17:34:16 -07001954 * Mark a child as having been dropped.
Patrick Dubroyde7658b2010-09-27 11:15:43 -07001955 * At the beginning of the drag operation, the child may have been on another
Patrick Dubroyce34a972010-10-19 10:34:32 -07001956 * screen, but it is re-parented before this method is called.
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001957 *
1958 * @param child The child that is being dropped
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001959 */
Adam Cohen716b51e2011-06-30 12:09:54 -07001960 void onDropChild(View child) {
Romain Guyd94533d2009-08-17 10:01:15 -07001961 if (child != null) {
Sebastian Francod4682992022-10-05 13:03:09 -05001962 CellLayoutLayoutParams
1963 lp = (CellLayoutLayoutParams) child.getLayoutParams();
Romain Guy84f296c2009-11-04 15:00:44 -08001964 lp.dropped = true;
Romain Guyd94533d2009-08-17 10:01:15 -07001965 child.requestLayout();
Tony Wickham1cdb6d02015-09-17 11:08:27 -07001966 markCellsAsOccupiedForView(child);
Romain Guyd94533d2009-08-17 10:01:15 -07001967 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001968 }
1969
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001970 /**
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001971 * Computes a bounding rectangle for a range of cells
Winson Chungaafa03c2010-06-11 17:34:16 -07001972 *
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001973 * @param cellX X coordinate of upper left corner expressed as a cell position
1974 * @param cellY Y coordinate of upper left corner expressed as a cell position
Winson Chungaafa03c2010-06-11 17:34:16 -07001975 * @param cellHSpan Width in cells
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001976 * @param cellVSpan Height in cells
Patrick Dubroy6569f2c2010-07-12 14:25:18 -07001977 * @param resultRect Rect into which to put the results
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001978 */
Adam Cohend41fbf52012-02-16 23:53:59 -08001979 public void cellToRect(int cellX, int cellY, int cellHSpan, int cellVSpan, Rect resultRect) {
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001980 final int cellWidth = mCellWidth;
1981 final int cellHeight = mCellHeight;
Winson Chungaafa03c2010-06-11 17:34:16 -07001982
Pierre Barbier de Reuille1b8bbb62021-05-19 22:45:16 +01001983 // We observe a shift of 1 pixel on the x coordinate compared to the actual cell coordinates
1984 final int hStartPadding = getPaddingLeft()
1985 + (int) Math.ceil(getUnusedHorizontalSpace() / 2f);
Winson Chung4b825dcd2011-06-19 12:41:22 -07001986 final int vStartPadding = getPaddingTop();
Winson Chungaafa03c2010-06-11 17:34:16 -07001987
Thales Lima78d00ad2021-09-30 11:29:06 +01001988 int x = hStartPadding + (cellX * mBorderSpace.x) + (cellX * cellWidth);
1989 int y = vStartPadding + (cellY * mBorderSpace.y) + (cellY * cellHeight);
Jon Miranda228877d2021-02-09 11:05:00 -05001990
Thales Lima78d00ad2021-09-30 11:29:06 +01001991 int width = cellHSpan * cellWidth + ((cellHSpan - 1) * mBorderSpace.x);
1992 int height = cellVSpan * cellHeight + ((cellVSpan - 1) * mBorderSpace.y);
Winson Chungaafa03c2010-06-11 17:34:16 -07001993
Patrick Dubroy6569f2c2010-07-12 14:25:18 -07001994 resultRect.set(x, y, x + width, y + height);
The Android Open Source Project31dd5032009-03-03 19:32:27 -08001995 }
Winson Chungaafa03c2010-06-11 17:34:16 -07001996
Adam Cohend4844c32011-02-18 19:25:06 -08001997 public void markCellsAsOccupiedForView(View view) {
Sebastian Francof153d912022-04-22 16:15:27 -05001998 if (view instanceof LauncherAppWidgetHostView
1999 && view.getTag() instanceof LauncherAppWidgetInfo) {
2000 LauncherAppWidgetInfo info = (LauncherAppWidgetInfo) view.getTag();
Sunny Goyal669b71f2023-01-27 14:37:07 -08002001 CellPos pos = mActivity.getCellPosMapper().mapModelToPresenter(info);
2002 mOccupied.markCells(pos.cellX, pos.cellY, info.spanX, info.spanY, true);
Sebastian Francof153d912022-04-22 16:15:27 -05002003 return;
2004 }
Michael Jurkaa52570f2012-03-20 03:18:20 -07002005 if (view == null || view.getParent() != mShortcutsAndWidgets) return;
Sebastian Francod4682992022-10-05 13:03:09 -05002006 CellLayoutLayoutParams
2007 lp = (CellLayoutLayoutParams) view.getLayoutParams();
Sebastian Franco877088e2023-01-03 15:16:22 -07002008 mOccupied.markCells(lp.getCellX(), lp.getCellY(), lp.cellHSpan, lp.cellVSpan, true);
Michael Jurka0280c3b2010-09-17 15:00:07 -07002009 }
2010
Adam Cohend4844c32011-02-18 19:25:06 -08002011 public void markCellsAsUnoccupiedForView(View view) {
Sebastian Francof153d912022-04-22 16:15:27 -05002012 if (view instanceof LauncherAppWidgetHostView
2013 && view.getTag() instanceof LauncherAppWidgetInfo) {
2014 LauncherAppWidgetInfo info = (LauncherAppWidgetInfo) view.getTag();
Sunny Goyal669b71f2023-01-27 14:37:07 -08002015 CellPos pos = mActivity.getCellPosMapper().mapModelToPresenter(info);
2016 mOccupied.markCells(pos.cellX, pos.cellY, info.spanX, info.spanY, false);
Sebastian Francof153d912022-04-22 16:15:27 -05002017 return;
2018 }
Michael Jurkaa52570f2012-03-20 03:18:20 -07002019 if (view == null || view.getParent() != mShortcutsAndWidgets) return;
Sebastian Francod4682992022-10-05 13:03:09 -05002020 CellLayoutLayoutParams
2021 lp = (CellLayoutLayoutParams) view.getLayoutParams();
Sebastian Franco877088e2023-01-03 15:16:22 -07002022 mOccupied.markCells(lp.getCellX(), lp.getCellY(), lp.cellHSpan, lp.cellVSpan, false);
The Android Open Source Project31dd5032009-03-03 19:32:27 -08002023 }
2024
Adam Cohen2801caf2011-05-13 20:57:39 -07002025 public int getDesiredWidth() {
Jon Miranda228877d2021-02-09 11:05:00 -05002026 return getPaddingLeft() + getPaddingRight() + (mCountX * mCellWidth)
Thales Lima78d00ad2021-09-30 11:29:06 +01002027 + ((mCountX - 1) * mBorderSpace.x);
Adam Cohen2801caf2011-05-13 20:57:39 -07002028 }
2029
2030 public int getDesiredHeight() {
Jon Miranda228877d2021-02-09 11:05:00 -05002031 return getPaddingTop() + getPaddingBottom() + (mCountY * mCellHeight)
Thales Lima78d00ad2021-09-30 11:29:06 +01002032 + ((mCountY - 1) * mBorderSpace.y);
Adam Cohen2801caf2011-05-13 20:57:39 -07002033 }
2034
Michael Jurka66d72172011-04-12 16:29:25 -07002035 public boolean isOccupied(int x, int y) {
Schneider Victor-tulias750c5af2023-07-26 10:16:04 -07002036 if (x >= 0 && x < mCountX && y >= 0 && y < mCountY) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07002037 return mOccupied.cells[x][y];
Schneider Victor-tulias750c5af2023-07-26 10:16:04 -07002038 }
2039 if (BuildConfig.IS_STUDIO_BUILD) {
Michael Jurka66d72172011-04-12 16:29:25 -07002040 throw new RuntimeException("Position exceeds the bound of this CellLayout");
2041 }
Schneider Victor-tulias750c5af2023-07-26 10:16:04 -07002042 return true;
Michael Jurka66d72172011-04-12 16:29:25 -07002043 }
2044
The Android Open Source Project31dd5032009-03-03 19:32:27 -08002045 @Override
2046 public ViewGroup.LayoutParams generateLayoutParams(AttributeSet attrs) {
Sebastian Francod4682992022-10-05 13:03:09 -05002047 return new CellLayoutLayoutParams(getContext(), attrs);
The Android Open Source Project31dd5032009-03-03 19:32:27 -08002048 }
2049
2050 @Override
2051 protected boolean checkLayoutParams(ViewGroup.LayoutParams p) {
Sebastian Francod4682992022-10-05 13:03:09 -05002052 return p instanceof CellLayoutLayoutParams;
The Android Open Source Project31dd5032009-03-03 19:32:27 -08002053 }
2054
2055 @Override
2056 protected ViewGroup.LayoutParams generateLayoutParams(ViewGroup.LayoutParams p) {
Sebastian Francod4682992022-10-05 13:03:09 -05002057 return new CellLayoutLayoutParams(p);
The Android Open Source Project31dd5032009-03-03 19:32:27 -08002058 }
2059
Samuel Fufa1e2d0042019-11-18 17:12:46 -08002060 /**
Tony Wickham86930612015-09-09 13:50:40 -07002061 * Returns whether an item can be placed in this CellLayout (after rearranging and/or resizing
2062 * if necessary).
2063 */
2064 public boolean hasReorderSolution(ItemInfo itemInfo) {
2065 int[] cellPoint = new int[2];
2066 // Check for a solution starting at every cell.
2067 for (int cellX = 0; cellX < getCountX(); cellX++) {
2068 for (int cellY = 0; cellY < getCountY(); cellY++) {
2069 cellToPoint(cellX, cellY, cellPoint);
2070 if (findReorderSolution(cellPoint[0], cellPoint[1], itemInfo.minSpanX,
2071 itemInfo.minSpanY, itemInfo.spanX, itemInfo.spanY, mDirectionVector, null,
2072 true, new ItemConfiguration()).isSolution) {
2073 return true;
2074 }
2075 }
2076 }
2077 return false;
2078 }
2079
Samuel Fufaa4211432020-02-25 18:47:54 -08002080 /**
2081 * Finds solution to accept hotseat migration to cell layout. commits solution if commitConfig
2082 */
2083 public boolean makeSpaceForHotseatMigration(boolean commitConfig) {
Samuel Fufaa4211432020-02-25 18:47:54 -08002084 int[] cellPoint = new int[2];
2085 int[] directionVector = new int[]{0, -1};
2086 cellToPoint(0, mCountY, cellPoint);
2087 ItemConfiguration configuration = new ItemConfiguration();
2088 if (findReorderSolution(cellPoint[0], cellPoint[1], mCountX, 1, mCountX, 1,
2089 directionVector, null, false, configuration).isSolution) {
2090 if (commitConfig) {
2091 copySolutionToTempState(configuration, null);
Sunny Goyal711c5962021-06-23 12:36:18 -07002092 commitTempPlacement(null);
Samuel Fufa82bbdac2020-03-09 18:24:47 -07002093 // undo marking cells occupied since there is actually nothing being placed yet.
2094 mOccupied.markCells(0, mCountY - 1, mCountX, 1, false);
Samuel Fufaa4211432020-02-25 18:47:54 -08002095 }
2096 return true;
2097 }
2098 return false;
2099 }
2100
Samuel Fufa82bbdac2020-03-09 18:24:47 -07002101 /**
2102 * returns a copy of cell layout's grid occupancy
2103 */
2104 public GridOccupancy cloneGridOccupancy() {
2105 GridOccupancy occupancy = new GridOccupancy(mCountX, mCountY);
2106 mOccupied.copyTo(occupancy);
2107 return occupancy;
2108 }
2109
Sunny Goyal9ca9c132015-04-29 14:57:22 -07002110 public boolean isRegionVacant(int x, int y, int spanX, int spanY) {
Sunny Goyalff4ba2d2016-04-02 14:12:34 -07002111 return mOccupied.isRegionVacant(x, y, spanX, spanY);
Sunny Goyal9ca9c132015-04-29 14:57:22 -07002112 }
Sebastian Franco9ea36d42023-09-21 13:56:42 -07002113
2114 public void setSpaceBetweenCellLayoutsPx(@Px int spaceBetweenCellLayoutsPx) {
2115 mSpaceBetweenCellLayoutsPx = spaceBetweenCellLayoutsPx;
2116 }
The Android Open Source Project31dd5032009-03-03 19:32:27 -08002117}