blob: 44403e2df1cd1fe256ad8a02f37ab1bcac9cbcd2 [file] [log] [blame]
Winson Chung97d85d22011-04-13 11:27:36 -07001/*
Hyunyoung Songee3e6a72015-02-20 14:25:27 -08002 * Copyright (C) 2015 The Android Open Source Project
Winson Chung97d85d22011-04-13 11:27:36 -07003 *
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;
Winson Chung97d85d22011-04-13 11:27:36 -070018
Hyunyoung Songee3e6a72015-02-20 14:25:27 -080019import android.util.Log;
Winson Chung97d85d22011-04-13 11:27:36 -070020import android.view.KeyEvent;
Sunny Goyalb3726d92014-08-20 16:58:17 -070021import android.view.SoundEffectConstants;
Winson Chung97d85d22011-04-13 11:27:36 -070022import android.view.View;
23import android.view.ViewGroup;
Winson Chung97d85d22011-04-13 11:27:36 -070024
Hyunyoung Songee3e6a72015-02-20 14:25:27 -080025import com.android.launcher3.util.FocusLogic;
Adam Cohen091440a2015-03-18 14:16:05 -070026import com.android.launcher3.util.Thunk;
Winson Chungfaa13252011-06-13 18:15:54 -070027
Winson Chung97d85d22011-04-13 11:27:36 -070028/**
Winson Chung4d279d92011-07-21 11:46:32 -070029 * A keyboard listener we set on all the workspace icons.
30 */
Adam Cohenac56cff2011-09-28 20:45:37 -070031class IconKeyEventListener implements View.OnKeyListener {
Hyunyoung Songee3e6a72015-02-20 14:25:27 -080032 @Override
Winson Chung4d279d92011-07-21 11:46:32 -070033 public boolean onKey(View v, int keyCode, KeyEvent event) {
Adam Cohenac56cff2011-09-28 20:45:37 -070034 return FocusHelper.handleIconKeyEvent(v, keyCode, event);
35 }
36}
37
38/**
Winson Chung3d503fb2011-07-13 17:25:49 -070039 * A keyboard listener we set on all the hotseat buttons.
Winson Chung4e6a9762011-05-09 11:56:34 -070040 */
Adam Cohenac56cff2011-09-28 20:45:37 -070041class HotseatIconKeyEventListener implements View.OnKeyListener {
Hyunyoung Songee3e6a72015-02-20 14:25:27 -080042 @Override
Winson Chung4e6a9762011-05-09 11:56:34 -070043 public boolean onKey(View v, int keyCode, KeyEvent event) {
Hyunyoung Songee3e6a72015-02-20 14:25:27 -080044 return FocusHelper.handleHotseatButtonKeyEvent(v, keyCode, event);
Winson Chung4e6a9762011-05-09 11:56:34 -070045 }
46}
47
Winson Chung97d85d22011-04-13 11:27:36 -070048public class FocusHelper {
Winson Chung97d85d22011-04-13 11:27:36 -070049
Hyunyoung Songee3e6a72015-02-20 14:25:27 -080050 private static final String TAG = "FocusHelper";
51 private static final boolean DEBUG = false;
52
Hyunyoung Songee3e6a72015-02-20 14:25:27 -080053 /**
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -070054 * Handles key events in paged folder.
Sunny Goyal290800b2015-03-05 11:33:33 -080055 */
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -070056 public static class PagedFolderKeyEventListener implements View.OnKeyListener {
Sunny Goyal290800b2015-03-05 11:33:33 -080057
58 private final Folder mFolder;
59
60 public PagedFolderKeyEventListener(Folder folder) {
61 mFolder = folder;
62 }
63
64 @Override
Hyunyoung Songada50982015-04-10 14:35:23 -070065 public boolean onKey(View v, int keyCode, KeyEvent e) {
66 boolean consume = FocusLogic.shouldConsume(keyCode);
67 if (e.getAction() == KeyEvent.ACTION_UP) {
68 return consume;
Sunny Goyal290800b2015-03-05 11:33:33 -080069 }
Hyunyoung Songada50982015-04-10 14:35:23 -070070 if (DEBUG) {
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -070071 Log.v(TAG, String.format("Handle ALL Folders keyevent=[%s].",
Hyunyoung Songada50982015-04-10 14:35:23 -070072 KeyEvent.keyCodeToString(keyCode)));
73 }
Sunny Goyal290800b2015-03-05 11:33:33 -080074
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -070075 if (!(v.getParent() instanceof ShortcutAndWidgetContainer)) {
Hyunyoung Songada50982015-04-10 14:35:23 -070076 if (LauncherAppState.isDogfoodBuild()) {
77 throw new IllegalStateException("Parent of the focused item is not supported.");
78 } else {
79 return false;
80 }
81 }
Hyunyoung Songee3e6a72015-02-20 14:25:27 -080082
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -070083 // Initialize variables.
84 final ShortcutAndWidgetContainer itemContainer = (ShortcutAndWidgetContainer) v.getParent();
85 final CellLayout cellLayout = (CellLayout) itemContainer.getParent();
86 final int countX = cellLayout.getCountX();
87 final int countY = cellLayout.getCountY();
Hyunyoung Songada50982015-04-10 14:35:23 -070088
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -070089 final int iconIndex = itemContainer.indexOfChild(v);
90 final FolderPagedView pagedView = (FolderPagedView) cellLayout.getParent();
91
92 final int pageIndex = pagedView.indexOfChild(cellLayout);
93 final int pageCount = pagedView.getPageCount();
Sunny Goyalc6205602015-05-21 20:46:33 -070094 final boolean isLayoutRtl = Utilities.isRtl(v.getResources());
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -070095
96 int[][] matrix = FocusLogic.createSparseMatrix(cellLayout);
Hyunyoung Songada50982015-04-10 14:35:23 -070097 // Process focus.
Adam Cohen2e6da152015-05-06 11:42:25 -070098 int newIconIndex = FocusLogic.handleKeyEvent(keyCode, countX,
Sunny Goyalc6205602015-05-21 20:46:33 -070099 countY, matrix, iconIndex, pageIndex, pageCount, isLayoutRtl);
Hyunyoung Songada50982015-04-10 14:35:23 -0700100 if (newIconIndex == FocusLogic.NOOP) {
101 handleNoopKey(keyCode, v);
102 return consume;
103 }
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700104 ShortcutAndWidgetContainer newParent = null;
105 View child = null;
106
Hyunyoung Songada50982015-04-10 14:35:23 -0700107 switch (newIconIndex) {
108 case FocusLogic.PREVIOUS_PAGE_RIGHT_COLUMN:
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700109 case FocusLogic.PREVIOUS_PAGE_LEFT_COLUMN:
110 newParent = getCellLayoutChildrenForIndex(pagedView, pageIndex - 1);
Hyunyoung Songada50982015-04-10 14:35:23 -0700111 if (newParent != null) {
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700112 int row = ((CellLayout.LayoutParams) v.getLayoutParams()).cellY;
113 pagedView.snapToPage(pageIndex - 1);
114 child = newParent.getChildAt(
115 ((newIconIndex == FocusLogic.PREVIOUS_PAGE_LEFT_COLUMN)
116 ^ newParent.invertLayoutHorizontally()) ? 0 : countX - 1, row);
Hyunyoung Songada50982015-04-10 14:35:23 -0700117 }
118 break;
119 case FocusLogic.PREVIOUS_PAGE_FIRST_ITEM:
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700120 newParent = getCellLayoutChildrenForIndex(pagedView, pageIndex - 1);
Hyunyoung Songada50982015-04-10 14:35:23 -0700121 if (newParent != null) {
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700122 pagedView.snapToPage(pageIndex - 1);
123 child = newParent.getChildAt(0, 0);
Hyunyoung Songada50982015-04-10 14:35:23 -0700124 }
125 break;
126 case FocusLogic.PREVIOUS_PAGE_LAST_ITEM:
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700127 newParent = getCellLayoutChildrenForIndex(pagedView, pageIndex - 1);
Hyunyoung Songada50982015-04-10 14:35:23 -0700128 if (newParent != null) {
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700129 pagedView.snapToPage(pageIndex - 1);
130 child = newParent.getChildAt(countX - 1, countY - 1);
Hyunyoung Songada50982015-04-10 14:35:23 -0700131 }
132 break;
133 case FocusLogic.NEXT_PAGE_FIRST_ITEM:
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700134 newParent = getCellLayoutChildrenForIndex(pagedView, pageIndex + 1);
Hyunyoung Songada50982015-04-10 14:35:23 -0700135 if (newParent != null) {
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700136 pagedView.snapToPage(pageIndex + 1);
137 child = newParent.getChildAt(0, 0);
Hyunyoung Songada50982015-04-10 14:35:23 -0700138 }
139 break;
140 case FocusLogic.NEXT_PAGE_LEFT_COLUMN:
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700141 case FocusLogic.NEXT_PAGE_RIGHT_COLUMN:
142 newParent = getCellLayoutChildrenForIndex(pagedView, pageIndex + 1);
Hyunyoung Songada50982015-04-10 14:35:23 -0700143 if (newParent != null) {
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700144 pagedView.snapToPage(pageIndex + 1);
Tony Wickham25189852015-11-04 17:44:32 -0800145 child = FocusLogic.getAdjacentChildInNextFolderPage(
146 newParent, v, newIconIndex);
Hyunyoung Songada50982015-04-10 14:35:23 -0700147 }
148 break;
149 case FocusLogic.CURRENT_PAGE_FIRST_ITEM:
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700150 child = cellLayout.getChildAt(0, 0);
Hyunyoung Songada50982015-04-10 14:35:23 -0700151 break;
152 case FocusLogic.CURRENT_PAGE_LAST_ITEM:
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700153 child = pagedView.getLastItem();
Hyunyoung Songada50982015-04-10 14:35:23 -0700154 break;
155 default: // Go to some item on the current page.
156 child = itemContainer.getChildAt(newIconIndex);
157 break;
158 }
159 if (child != null) {
160 child.requestFocus();
161 playSoundEffect(keyCode, v);
162 } else {
163 handleNoopKey(keyCode, v);
164 }
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800165 return consume;
166 }
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800167
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700168 public void handleNoopKey(int keyCode, View v) {
169 if (keyCode == KeyEvent.KEYCODE_DPAD_DOWN) {
170 mFolder.mFolderName.requestFocus();
171 playSoundEffect(keyCode, v);
172 }
173 }
Sunny Goyal290800b2015-03-05 11:33:33 -0800174 }
175
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800176 /**
Tony Wickham4fc82872015-11-10 16:52:14 -0800177 * Handles key events in the workspace hotseat (bottom of the screen).
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800178 * <p>Currently we don't special case for the phone UI in different orientations, even though
179 * the hotseat is on the side in landscape mode. This is to ensure that accessibility
180 * consistency is maintained across rotations.
181 */
182 static boolean handleHotseatButtonKeyEvent(View v, int keyCode, KeyEvent e) {
183 boolean consume = FocusLogic.shouldConsume(keyCode);
184 if (e.getAction() == KeyEvent.ACTION_UP || !consume) {
185 return consume;
186 }
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800187
Winsonc0b52fe2015-09-09 16:38:15 -0700188 final Launcher launcher = (Launcher) v.getContext();
189 final DeviceProfile profile = launcher.getDeviceProfile();
Adam Cohen2e6da152015-05-06 11:42:25 -0700190
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800191 if (DEBUG) {
192 Log.v(TAG, String.format(
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700193 "Handle HOTSEAT BUTTONS keyevent=[%s] on hotseat buttons, isVertical=%s",
194 KeyEvent.keyCodeToString(keyCode), profile.isVerticalBarLayout()));
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800195 }
196
197 // Initialize the variables.
Winsonfa56b3f2015-09-14 12:01:13 -0700198 final Workspace workspace = (Workspace) v.getRootView().findViewById(R.id.workspace);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800199 final ShortcutAndWidgetContainer hotseatParent = (ShortcutAndWidgetContainer) v.getParent();
200 final CellLayout hotseatLayout = (CellLayout) hotseatParent.getParent();
Hyunyoung Song31178b82015-02-24 14:12:51 -0800201
Winsonfa56b3f2015-09-14 12:01:13 -0700202 final ItemInfo itemInfo = (ItemInfo) v.getTag();
Hyunyoung Songb76cd622015-04-16 14:34:09 -0700203 int pageIndex = workspace.getNextPage();
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800204 int pageCount = workspace.getChildCount();
Hyunyoung Song31178b82015-02-24 14:12:51 -0800205 int countX = -1;
206 int countY = -1;
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700207 int iconIndex = hotseatParent.indexOfChild(v);
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700208 int iconRank = ((CellLayout.LayoutParams) hotseatLayout.getShortcutsAndWidgets()
209 .getChildAt(iconIndex).getLayoutParams()).cellX;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800210
211 final CellLayout iconLayout = (CellLayout) workspace.getChildAt(pageIndex);
Hyunyoung Songb76cd622015-04-16 14:34:09 -0700212 if (iconLayout == null) {
213 // This check is to guard against cases where key strokes rushes in when workspace
214 // child creation/deletion is still in flux. (e.g., during drop or fling
215 // animation.)
216 return consume;
217 }
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800218 final ViewGroup iconParent = iconLayout.getShortcutsAndWidgets();
219
220 ViewGroup parent = null;
Hyunyoung Song31178b82015-02-24 14:12:51 -0800221 int[][] matrix = null;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800222
223 if (keyCode == KeyEvent.KEYCODE_DPAD_UP &&
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700224 !profile.isVerticalBarLayout()) {
225 matrix = FocusLogic.createSparseMatrix(iconLayout, hotseatLayout,
Sunny Goyal4f3e9382015-06-05 00:13:25 -0700226 true /* hotseat horizontal */, profile.inv.hotseatAllAppsRank,
227 iconRank == profile.inv.hotseatAllAppsRank /* include all apps icon */);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800228 iconIndex += iconParent.getChildCount();
229 countX = iconLayout.getCountX();
230 countY = iconLayout.getCountY() + hotseatLayout.getCountY();
231 parent = iconParent;
232 } else if (keyCode == KeyEvent.KEYCODE_DPAD_LEFT &&
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700233 profile.isVerticalBarLayout()) {
234 matrix = FocusLogic.createSparseMatrix(iconLayout, hotseatLayout,
Sunny Goyal4f3e9382015-06-05 00:13:25 -0700235 false /* hotseat horizontal */, profile.inv.hotseatAllAppsRank,
236 iconRank == profile.inv.hotseatAllAppsRank /* include all apps icon */);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800237 iconIndex += iconParent.getChildCount();
238 countX = iconLayout.getCountX() + hotseatLayout.getCountX();
239 countY = iconLayout.getCountY();
240 parent = iconParent;
Hyunyoung Song31178b82015-02-24 14:12:51 -0800241 } else if (keyCode == KeyEvent.KEYCODE_DPAD_RIGHT &&
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700242 profile.isVerticalBarLayout()) {
Hyunyoung Song31178b82015-02-24 14:12:51 -0800243 keyCode = KeyEvent.KEYCODE_PAGE_DOWN;
Winsonfa56b3f2015-09-14 12:01:13 -0700244 } else if (isUninstallKeyChord(e)) {
245 matrix = FocusLogic.createSparseMatrix(iconLayout);
246 if (UninstallDropTarget.supportsDrop(launcher, itemInfo)) {
247 UninstallDropTarget.startUninstallActivity(launcher, itemInfo);
248 }
249 } else if (isDeleteKeyChord(e)) {
250 matrix = FocusLogic.createSparseMatrix(iconLayout);
Winson2949fb52015-09-24 09:56:11 -0700251 launcher.removeItem(v, itemInfo, true /* deleteFromDb */);
Winsonc0b52fe2015-09-09 16:38:15 -0700252 } else {
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800253 // For other KEYCODE_DPAD_LEFT and KEYCODE_DPAD_RIGHT navigation, do not use the
254 // matrix extended with hotseat.
255 matrix = FocusLogic.createSparseMatrix(hotseatLayout);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800256 countX = hotseatLayout.getCountX();
257 countY = hotseatLayout.getCountY();
Hyunyoung Song31178b82015-02-24 14:12:51 -0800258 parent = hotseatParent;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800259 }
260
261 // Process the focus.
Adam Cohen2e6da152015-05-06 11:42:25 -0700262 int newIconIndex = FocusLogic.handleKeyEvent(keyCode, countX,
Sunny Goyalc6205602015-05-21 20:46:33 -0700263 countY, matrix, iconIndex, pageIndex, pageCount, Utilities.isRtl(v.getResources()));
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800264
Hyunyoung Song31178b82015-02-24 14:12:51 -0800265 View newIcon = null;
Tony Wickham4fc82872015-11-10 16:52:14 -0800266 switch (newIconIndex) {
267 case FocusLogic.NEXT_PAGE_FIRST_ITEM:
268 parent = getCellLayoutChildrenForIndex(workspace, pageIndex + 1);
269 newIcon = parent.getChildAt(0);
270 // TODO(hyunyoungs): handle cases where the child is not an icon but
271 // a folder or a widget.
272 workspace.snapToPage(pageIndex + 1);
273 break;
274 case FocusLogic.PREVIOUS_PAGE_FIRST_ITEM:
275 parent = getCellLayoutChildrenForIndex(workspace, pageIndex - 1);
276 newIcon = parent.getChildAt(0);
277 // TODO(hyunyoungs): handle cases where the child is not an icon but
278 // a folder or a widget.
279 workspace.snapToPage(pageIndex - 1);
280 break;
281 case FocusLogic.PREVIOUS_PAGE_LAST_ITEM:
282 parent = getCellLayoutChildrenForIndex(workspace, pageIndex - 1);
283 newIcon = parent.getChildAt(parent.getChildCount() - 1);
284 // TODO(hyunyoungs): handle cases where the child is not an icon but
285 // a folder or a widget.
286 workspace.snapToPage(pageIndex - 1);
287 break;
288 case FocusLogic.PREVIOUS_PAGE_LEFT_COLUMN:
289 case FocusLogic.PREVIOUS_PAGE_RIGHT_COLUMN:
290 // Go to the previous page but keep the focus on the same hotseat icon.
291 workspace.snapToPage(pageIndex - 1);
292 break;
293 case FocusLogic.NEXT_PAGE_LEFT_COLUMN:
294 case FocusLogic.NEXT_PAGE_RIGHT_COLUMN:
295 // Go to the next page but keep the focus on the same hotseat icon.
296 workspace.snapToPage(pageIndex + 1);
297 break;
Hyunyoung Song31178b82015-02-24 14:12:51 -0800298 }
299 if (parent == iconParent && newIconIndex >= iconParent.getChildCount()) {
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800300 newIconIndex -= iconParent.getChildCount();
301 }
302 if (parent != null) {
Hyunyoung Song31178b82015-02-24 14:12:51 -0800303 if (newIcon == null && newIconIndex >=0) {
304 newIcon = parent.getChildAt(newIconIndex);
305 }
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800306 if (newIcon != null) {
307 newIcon.requestFocus();
308 playSoundEffect(keyCode, v);
309 }
310 }
311 return consume;
312 }
313
314 /**
315 * Handles key events in a workspace containing icons.
316 */
317 static boolean handleIconKeyEvent(View v, int keyCode, KeyEvent e) {
318 boolean consume = FocusLogic.shouldConsume(keyCode);
319 if (e.getAction() == KeyEvent.ACTION_UP || !consume) {
320 return consume;
321 }
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700322
Adam Cohen2e6da152015-05-06 11:42:25 -0700323 Launcher launcher = (Launcher) v.getContext();
324 DeviceProfile profile = launcher.getDeviceProfile();
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700325
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800326 if (DEBUG) {
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700327 Log.v(TAG, String.format("Handle WORKSPACE ICONS keyevent=[%s] isVerticalBar=%s",
328 KeyEvent.keyCodeToString(keyCode), profile.isVerticalBarLayout()));
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800329 }
330
331 // Initialize the variables.
332 ShortcutAndWidgetContainer parent = (ShortcutAndWidgetContainer) v.getParent();
Hyunyoung Song38531712015-03-03 19:25:16 -0800333 CellLayout iconLayout = (CellLayout) parent.getParent();
Hyunyoung Song31178b82015-02-24 14:12:51 -0800334 final Workspace workspace = (Workspace) iconLayout.getParent();
Adam Cohen2e6da152015-05-06 11:42:25 -0700335 final ViewGroup dragLayer = (ViewGroup) workspace.getParent();
336 final ViewGroup tabs = (ViewGroup) dragLayer.findViewById(R.id.search_drop_target_bar);
337 final Hotseat hotseat = (Hotseat) dragLayer.findViewById(R.id.hotseat);
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700338
Winsonfa56b3f2015-09-14 12:01:13 -0700339 final ItemInfo itemInfo = (ItemInfo) v.getTag();
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700340 final int iconIndex = parent.indexOfChild(v);
341 final int pageIndex = workspace.indexOfChild(iconLayout);
342 final int pageCount = workspace.getChildCount();
Hyunyoung Song31178b82015-02-24 14:12:51 -0800343 int countX = iconLayout.getCountX();
344 int countY = iconLayout.getCountY();
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800345
346 CellLayout hotseatLayout = (CellLayout) hotseat.getChildAt(0);
347 ShortcutAndWidgetContainer hotseatParent = hotseatLayout.getShortcutsAndWidgets();
348 int[][] matrix;
349
Hyunyoung Song31178b82015-02-24 14:12:51 -0800350 // KEYCODE_DPAD_DOWN in portrait (KEYCODE_DPAD_RIGHT in landscape) is the only key allowed
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800351 // to take a user to the hotseat. For other dpad navigation, do not use the matrix extended
352 // with the hotseat.
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700353 if (keyCode == KeyEvent.KEYCODE_DPAD_DOWN && !profile.isVerticalBarLayout()) {
354 matrix = FocusLogic.createSparseMatrix(iconLayout, hotseatLayout, true /* horizontal */,
Sunny Goyal4f3e9382015-06-05 00:13:25 -0700355 profile.inv.hotseatAllAppsRank,
Winson Chungc393b072015-05-20 15:03:13 -0700356 !hotseat.hasIcons() /* ignore all apps icon, unless there are no other icons */);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800357 countY = countY + 1;
Hyunyoung Song31178b82015-02-24 14:12:51 -0800358 } else if (keyCode == KeyEvent.KEYCODE_DPAD_RIGHT &&
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700359 profile.isVerticalBarLayout()) {
360 matrix = FocusLogic.createSparseMatrix(iconLayout, hotseatLayout, false /* horizontal */,
Sunny Goyal4f3e9382015-06-05 00:13:25 -0700361 profile.inv.hotseatAllAppsRank,
Winson Chungc393b072015-05-20 15:03:13 -0700362 !hotseat.hasIcons() /* ignore all apps icon, unless there are no other icons */);
Hyunyoung Song31178b82015-02-24 14:12:51 -0800363 countX = countX + 1;
Winsonfa56b3f2015-09-14 12:01:13 -0700364 } else if (isUninstallKeyChord(e)) {
365 matrix = FocusLogic.createSparseMatrix(iconLayout);
366 if (UninstallDropTarget.supportsDrop(launcher, itemInfo)) {
367 UninstallDropTarget.startUninstallActivity(launcher, itemInfo);
368 }
369 } else if (isDeleteKeyChord(e)) {
370 matrix = FocusLogic.createSparseMatrix(iconLayout);
Winson2949fb52015-09-24 09:56:11 -0700371 launcher.removeItem(v, itemInfo, true /* deleteFromDb */);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800372 } else {
Hyunyoung Song31178b82015-02-24 14:12:51 -0800373 matrix = FocusLogic.createSparseMatrix(iconLayout);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800374 }
375
376 // Process the focus.
Adam Cohen2e6da152015-05-06 11:42:25 -0700377 int newIconIndex = FocusLogic.handleKeyEvent(keyCode, countX,
Sunny Goyalc6205602015-05-21 20:46:33 -0700378 countY, matrix, iconIndex, pageIndex, pageCount, Utilities.isRtl(v.getResources()));
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800379 View newIcon = null;
380 switch (newIconIndex) {
381 case FocusLogic.NOOP:
382 if (keyCode == KeyEvent.KEYCODE_DPAD_UP) {
383 newIcon = tabs;
384 }
385 break;
Hyunyoung Song38531712015-03-03 19:25:16 -0800386 case FocusLogic.PREVIOUS_PAGE_RIGHT_COLUMN:
Hyunyoung Songada50982015-04-10 14:35:23 -0700387 case FocusLogic.NEXT_PAGE_RIGHT_COLUMN:
388 int newPageIndex = pageIndex - 1;
389 if (newIconIndex == FocusLogic.NEXT_PAGE_RIGHT_COLUMN) {
390 newPageIndex = pageIndex + 1;
391 }
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700392 int row = ((CellLayout.LayoutParams) v.getLayoutParams()).cellY;
Hyunyoung Songada50982015-04-10 14:35:23 -0700393 parent = getCellLayoutChildrenForIndex(workspace, newPageIndex);
Hyunyoung Song38531712015-03-03 19:25:16 -0800394 if (parent != null) {
395 iconLayout = (CellLayout) parent.getParent();
Hyunyoung Songac721f82015-03-04 16:33:56 -0800396 matrix = FocusLogic.createSparseMatrix(iconLayout,
Tony Wickham4fc82872015-11-10 16:52:14 -0800397 iconLayout.getCountX(), row);
Adam Cohen2e6da152015-05-06 11:42:25 -0700398 newIconIndex = FocusLogic.handleKeyEvent(keyCode, countX + 1, countY,
Sunny Goyalc6205602015-05-21 20:46:33 -0700399 matrix, FocusLogic.PIVOT, newPageIndex, pageCount,
400 Utilities.isRtl(v.getResources()));
Hyunyoung Song38531712015-03-03 19:25:16 -0800401 newIcon = parent.getChildAt(newIconIndex);
402 }
403 break;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800404 case FocusLogic.PREVIOUS_PAGE_FIRST_ITEM:
405 parent = getCellLayoutChildrenForIndex(workspace, pageIndex - 1);
406 newIcon = parent.getChildAt(0);
Hyunyoung Song38531712015-03-03 19:25:16 -0800407 break;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800408 case FocusLogic.PREVIOUS_PAGE_LAST_ITEM:
409 parent = getCellLayoutChildrenForIndex(workspace, pageIndex - 1);
410 newIcon = parent.getChildAt(parent.getChildCount() - 1);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800411 break;
412 case FocusLogic.NEXT_PAGE_FIRST_ITEM:
413 parent = getCellLayoutChildrenForIndex(workspace, pageIndex + 1);
414 newIcon = parent.getChildAt(0);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800415 break;
Hyunyoung Song38531712015-03-03 19:25:16 -0800416 case FocusLogic.NEXT_PAGE_LEFT_COLUMN:
Hyunyoung Songada50982015-04-10 14:35:23 -0700417 case FocusLogic.PREVIOUS_PAGE_LEFT_COLUMN:
418 newPageIndex = pageIndex + 1;
419 if (newIconIndex == FocusLogic.PREVIOUS_PAGE_LEFT_COLUMN) {
420 newPageIndex = pageIndex - 1;
421 }
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700422 row = ((CellLayout.LayoutParams) v.getLayoutParams()).cellY;
Hyunyoung Songada50982015-04-10 14:35:23 -0700423 parent = getCellLayoutChildrenForIndex(workspace, newPageIndex);
Hyunyoung Song38531712015-03-03 19:25:16 -0800424 if (parent != null) {
425 iconLayout = (CellLayout) parent.getParent();
Hyunyoung Songac721f82015-03-04 16:33:56 -0800426 matrix = FocusLogic.createSparseMatrix(iconLayout, -1, row);
Adam Cohen2e6da152015-05-06 11:42:25 -0700427 newIconIndex = FocusLogic.handleKeyEvent(keyCode, countX + 1, countY,
Sunny Goyalc6205602015-05-21 20:46:33 -0700428 matrix, FocusLogic.PIVOT, newPageIndex, pageCount,
429 Utilities.isRtl(v.getResources()));
Hyunyoung Song38531712015-03-03 19:25:16 -0800430 newIcon = parent.getChildAt(newIconIndex);
431 }
432 break;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800433 case FocusLogic.CURRENT_PAGE_FIRST_ITEM:
434 newIcon = parent.getChildAt(0);
435 break;
436 case FocusLogic.CURRENT_PAGE_LAST_ITEM:
437 newIcon = parent.getChildAt(parent.getChildCount() - 1);
438 break;
439 default:
440 // current page, some item.
441 if (0 <= newIconIndex && newIconIndex < parent.getChildCount()) {
442 newIcon = parent.getChildAt(newIconIndex);
443 } else if (parent.getChildCount() <= newIconIndex &&
444 newIconIndex < parent.getChildCount() + hotseatParent.getChildCount()) {
445 newIcon = hotseatParent.getChildAt(newIconIndex - parent.getChildCount());
446 }
447 break;
448 }
449 if (newIcon != null) {
450 newIcon.requestFocus();
451 playSoundEffect(keyCode, v);
452 }
453 return consume;
454 }
455
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800456 //
457 // Helper methods.
458 //
459
Winson Chung97d85d22011-04-13 11:27:36 -0700460 /**
Winson Chung97d85d22011-04-13 11:27:36 -0700461 * Private helper method to get the CellLayoutChildren given a CellLayout index.
462 */
Sunny Goyal316490e2015-06-02 09:38:28 -0700463 @Thunk static ShortcutAndWidgetContainer getCellLayoutChildrenForIndex(
Michael Jurkaa52570f2012-03-20 03:18:20 -0700464 ViewGroup container, int i) {
Sunny Goyalb3726d92014-08-20 16:58:17 -0700465 CellLayout parent = (CellLayout) container.getChildAt(i);
466 return parent.getShortcutsAndWidgets();
Winson Chung97d85d22011-04-13 11:27:36 -0700467 }
468
Winson Chung97d85d22011-04-13 11:27:36 -0700469 /**
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800470 * Helper method to be used for playing sound effects.
Winson Chung97d85d22011-04-13 11:27:36 -0700471 */
Adam Cohen091440a2015-03-18 14:16:05 -0700472 @Thunk static void playSoundEffect(int keyCode, View v) {
Winson Chung97d85d22011-04-13 11:27:36 -0700473 switch (keyCode) {
474 case KeyEvent.KEYCODE_DPAD_LEFT:
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800475 v.playSoundEffect(SoundEffectConstants.NAVIGATION_LEFT);
Winson Chung97d85d22011-04-13 11:27:36 -0700476 break;
477 case KeyEvent.KEYCODE_DPAD_RIGHT:
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800478 v.playSoundEffect(SoundEffectConstants.NAVIGATION_RIGHT);
Winson Chung97d85d22011-04-13 11:27:36 -0700479 break;
480 case KeyEvent.KEYCODE_DPAD_DOWN:
Winson Chung97d85d22011-04-13 11:27:36 -0700481 case KeyEvent.KEYCODE_PAGE_DOWN:
Winson Chung97d85d22011-04-13 11:27:36 -0700482 case KeyEvent.KEYCODE_MOVE_END:
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800483 v.playSoundEffect(SoundEffectConstants.NAVIGATION_DOWN);
Adam Cohenac56cff2011-09-28 20:45:37 -0700484 break;
485 case KeyEvent.KEYCODE_DPAD_UP:
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800486 case KeyEvent.KEYCODE_PAGE_UP:
Adam Cohenac56cff2011-09-28 20:45:37 -0700487 case KeyEvent.KEYCODE_MOVE_HOME:
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800488 v.playSoundEffect(SoundEffectConstants.NAVIGATION_UP);
Adam Cohenac56cff2011-09-28 20:45:37 -0700489 break;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800490 default:
Winson Chung97d85d22011-04-13 11:27:36 -0700491 break;
Winson Chung97d85d22011-04-13 11:27:36 -0700492 }
Winson Chung97d85d22011-04-13 11:27:36 -0700493 }
Winsonfa56b3f2015-09-14 12:01:13 -0700494
495 /**
496 * Returns whether the key event represents a valid uninstall key chord.
497 */
498 private static boolean isUninstallKeyChord(KeyEvent event) {
499 int keyCode = event.getKeyCode();
500 return (keyCode == KeyEvent.KEYCODE_DEL || keyCode == KeyEvent.KEYCODE_FORWARD_DEL) &&
501 event.hasModifiers(KeyEvent.META_CTRL_ON | KeyEvent.META_SHIFT_ON);
502 }
503
504 /**
505 * Returns whether the key event represents a valid delete key chord.
506 */
507 private static boolean isDeleteKeyChord(KeyEvent event) {
508 int keyCode = event.getKeyCode();
509 return (keyCode == KeyEvent.KEYCODE_DEL || keyCode == KeyEvent.KEYCODE_FORWARD_DEL) &&
510 event.hasModifiers(KeyEvent.META_CTRL_ON);
511 }
Winson Chung97d85d22011-04-13 11:27:36 -0700512}