/* * Copyright (C) 2018 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.launcher3; import android.util.Log; import android.view.View; import android.view.ViewGroup; import com.android.launcher3.celllayout.CellLayoutLayoutParams; import com.android.launcher3.celllayout.CellPosMapper; import com.android.launcher3.celllayout.CellPosMapper.CellPos; import com.android.launcher3.folder.Folder; import com.android.launcher3.folder.FolderIcon; import com.android.launcher3.model.data.ItemInfo; import com.android.launcher3.touch.ItemLongClickListener; import com.android.launcher3.util.IntSet; public interface WorkspaceLayoutManager { String TAG = "Launcher.Workspace"; // The screen id used for the empty screen always present at the end. int EXTRA_EMPTY_SCREEN_ID = -201; // The screen id used for the second empty screen always present at the end for two panel home. int EXTRA_EMPTY_SCREEN_SECOND_ID = -200; // The screen ids used for the empty screens at the end of the workspaces. IntSet EXTRA_EMPTY_SCREEN_IDS = IntSet.wrap(EXTRA_EMPTY_SCREEN_ID, EXTRA_EMPTY_SCREEN_SECOND_ID); // The is the first screen. It is always present, even if its empty. int FIRST_SCREEN_ID = 0; // This is the second page. On two panel home it is always present, even if its empty. int SECOND_SCREEN_ID = 1; /** * At bind time, we use the rank (screenId) to compute x and y for hotseat items. * See {@link #addInScreen}. */ default void addInScreenFromBind(View child, ItemInfo info) { CellPos presenterPos = getCellPosMapper().mapModelToPresenter(info); int x = presenterPos.cellX; int y = presenterPos.cellY; if (info.container == LauncherSettings.Favorites.CONTAINER_HOTSEAT || info.container == LauncherSettings.Favorites.CONTAINER_HOTSEAT_PREDICTION) { int screenId = presenterPos.screenId; x = getHotseat().getCellXFromOrder(screenId); y = getHotseat().getCellYFromOrder(screenId); // TODO(b/335141365): Remove this log after the bug is fixed. Log.d(TAG, "addInScreenFromBind: hotseat inflation with x = " + x + " and y = " + y); } addInScreen(child, info.container, presenterPos.screenId, x, y, info.spanX, info.spanY); } /** * Adds the specified child in the specified screen based on the {@param info} * See {@link #addInScreen(View, int, int, int, int, int, int)}. */ default void addInScreen(View child, ItemInfo info) { CellPos presenterPos = getCellPosMapper().mapModelToPresenter(info); addInScreen(child, info.container, presenterPos.screenId, presenterPos.cellX, presenterPos.cellY, info.spanX, info.spanY); } /** * Adds the specified child in the specified screen. The position and dimension of * the child are defined by x, y, spanX and spanY. * * @param child The child to add in one of the workspace's screens. * @param screenId The screen in which to add the child. * @param x The X position of the child in the screen's grid. * @param y The Y position of the child in the screen's grid. * @param spanX The number of cells spanned horizontally by the child. * @param spanY The number of cells spanned vertically by the child. */ default void addInScreen(View child, int container, int screenId, int x, int y, int spanX, int spanY) { if (container == LauncherSettings.Favorites.CONTAINER_DESKTOP) { if (getScreenWithId(screenId) == null) { Log.e(TAG, "Skipping child, screenId " + screenId + " not found"); // DEBUGGING - Print out the stack trace to see where we are adding from new Throwable().printStackTrace(); return; } } if (EXTRA_EMPTY_SCREEN_IDS.contains(screenId)) { // This should never happen throw new RuntimeException("Screen id should not be extra empty screen: " + screenId); } final CellLayout layout; if (container == LauncherSettings.Favorites.CONTAINER_HOTSEAT || container == LauncherSettings.Favorites.CONTAINER_HOTSEAT_PREDICTION) { layout = getHotseat(); // Hide folder title in the hotseat if (child instanceof FolderIcon) { ((FolderIcon) child).setTextVisible(false); } } else { // Show folder title if not in the hotseat if (child instanceof FolderIcon) { ((FolderIcon) child).setTextVisible(true); } layout = getScreenWithId(screenId); } ViewGroup.LayoutParams genericLp = child.getLayoutParams(); CellLayoutLayoutParams lp; if (genericLp == null || !(genericLp instanceof CellLayoutLayoutParams)) { lp = new CellLayoutLayoutParams(x, y, spanX, spanY); } else { lp = (CellLayoutLayoutParams) genericLp; lp.setCellX(x); lp.setCellY(y); lp.cellHSpan = spanX; lp.cellVSpan = spanY; } if (spanX < 0 && spanY < 0) { lp.isLockedToGrid = false; } // Get the canonical child id to uniquely represent this view in this screen ItemInfo info = (ItemInfo) child.getTag(); int childId = info.getViewId(); boolean markCellsAsOccupied = !(child instanceof Folder); if (!layout.addViewToCellLayout(child, -1, childId, lp, markCellsAsOccupied)) { // TODO: This branch occurs when the workspace is adding views // outside of the defined grid // maybe we should be deleting these items from the LauncherModel? Log.e(TAG, "Failed to add to item at (" + lp.getCellX() + "," + lp.getCellY() + ") to CellLayout"); } child.setHapticFeedbackEnabled(false); child.setOnLongClickListener(getWorkspaceChildOnLongClickListener()); if (child instanceof DropTarget) { onAddDropTarget((DropTarget) child); } } default View.OnLongClickListener getWorkspaceChildOnLongClickListener() { return ItemLongClickListener.INSTANCE_WORKSPACE; } /** * Returns the mapper for converting between model and presenter */ CellPosMapper getCellPosMapper(); Hotseat getHotseat(); CellLayout getScreenWithId(int screenId); default void onAddDropTarget(DropTarget target) { } }