blob: 4709488a0107840c47e2c1b1cbbbe23e17c5591e [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,
Tony Wickham6cbd2222015-11-09 17:51:08 -0800226 true /* hotseat horizontal */, profile.inv.hotseatAllAppsRank);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800227 iconIndex += iconParent.getChildCount();
Tony Wickham6cbd2222015-11-09 17:51:08 -0800228 countX = hotseatLayout.getCountX();
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800229 countY = iconLayout.getCountY() + hotseatLayout.getCountY();
230 parent = iconParent;
231 } else if (keyCode == KeyEvent.KEYCODE_DPAD_LEFT &&
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700232 profile.isVerticalBarLayout()) {
233 matrix = FocusLogic.createSparseMatrix(iconLayout, hotseatLayout,
Tony Wickham6cbd2222015-11-09 17:51:08 -0800234 false /* hotseat horizontal */, profile.inv.hotseatAllAppsRank);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800235 iconIndex += iconParent.getChildCount();
236 countX = iconLayout.getCountX() + hotseatLayout.getCountX();
Tony Wickham6cbd2222015-11-09 17:51:08 -0800237 countY = hotseatLayout.getCountY();
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800238 parent = iconParent;
Hyunyoung Song31178b82015-02-24 14:12:51 -0800239 } else if (keyCode == KeyEvent.KEYCODE_DPAD_RIGHT &&
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700240 profile.isVerticalBarLayout()) {
Hyunyoung Song31178b82015-02-24 14:12:51 -0800241 keyCode = KeyEvent.KEYCODE_PAGE_DOWN;
Winsonfa56b3f2015-09-14 12:01:13 -0700242 } else if (isUninstallKeyChord(e)) {
243 matrix = FocusLogic.createSparseMatrix(iconLayout);
244 if (UninstallDropTarget.supportsDrop(launcher, itemInfo)) {
245 UninstallDropTarget.startUninstallActivity(launcher, itemInfo);
246 }
247 } else if (isDeleteKeyChord(e)) {
248 matrix = FocusLogic.createSparseMatrix(iconLayout);
Winson2949fb52015-09-24 09:56:11 -0700249 launcher.removeItem(v, itemInfo, true /* deleteFromDb */);
Winsonc0b52fe2015-09-09 16:38:15 -0700250 } else {
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800251 // For other KEYCODE_DPAD_LEFT and KEYCODE_DPAD_RIGHT navigation, do not use the
252 // matrix extended with hotseat.
253 matrix = FocusLogic.createSparseMatrix(hotseatLayout);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800254 countX = hotseatLayout.getCountX();
255 countY = hotseatLayout.getCountY();
Hyunyoung Song31178b82015-02-24 14:12:51 -0800256 parent = hotseatParent;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800257 }
258
259 // Process the focus.
Adam Cohen2e6da152015-05-06 11:42:25 -0700260 int newIconIndex = FocusLogic.handleKeyEvent(keyCode, countX,
Sunny Goyalc6205602015-05-21 20:46:33 -0700261 countY, matrix, iconIndex, pageIndex, pageCount, Utilities.isRtl(v.getResources()));
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800262
Hyunyoung Song31178b82015-02-24 14:12:51 -0800263 View newIcon = null;
Tony Wickham4fc82872015-11-10 16:52:14 -0800264 switch (newIconIndex) {
265 case FocusLogic.NEXT_PAGE_FIRST_ITEM:
266 parent = getCellLayoutChildrenForIndex(workspace, pageIndex + 1);
267 newIcon = parent.getChildAt(0);
268 // TODO(hyunyoungs): handle cases where the child is not an icon but
269 // a folder or a widget.
270 workspace.snapToPage(pageIndex + 1);
271 break;
272 case FocusLogic.PREVIOUS_PAGE_FIRST_ITEM:
273 parent = getCellLayoutChildrenForIndex(workspace, pageIndex - 1);
274 newIcon = parent.getChildAt(0);
275 // TODO(hyunyoungs): handle cases where the child is not an icon but
276 // a folder or a widget.
277 workspace.snapToPage(pageIndex - 1);
278 break;
279 case FocusLogic.PREVIOUS_PAGE_LAST_ITEM:
280 parent = getCellLayoutChildrenForIndex(workspace, pageIndex - 1);
281 newIcon = parent.getChildAt(parent.getChildCount() - 1);
282 // TODO(hyunyoungs): handle cases where the child is not an icon but
283 // a folder or a widget.
284 workspace.snapToPage(pageIndex - 1);
285 break;
286 case FocusLogic.PREVIOUS_PAGE_LEFT_COLUMN:
287 case FocusLogic.PREVIOUS_PAGE_RIGHT_COLUMN:
288 // Go to the previous page but keep the focus on the same hotseat icon.
289 workspace.snapToPage(pageIndex - 1);
290 break;
291 case FocusLogic.NEXT_PAGE_LEFT_COLUMN:
292 case FocusLogic.NEXT_PAGE_RIGHT_COLUMN:
293 // Go to the next page but keep the focus on the same hotseat icon.
294 workspace.snapToPage(pageIndex + 1);
295 break;
Hyunyoung Song31178b82015-02-24 14:12:51 -0800296 }
297 if (parent == iconParent && newIconIndex >= iconParent.getChildCount()) {
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800298 newIconIndex -= iconParent.getChildCount();
299 }
300 if (parent != null) {
Hyunyoung Song31178b82015-02-24 14:12:51 -0800301 if (newIcon == null && newIconIndex >=0) {
302 newIcon = parent.getChildAt(newIconIndex);
303 }
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800304 if (newIcon != null) {
305 newIcon.requestFocus();
306 playSoundEffect(keyCode, v);
307 }
308 }
309 return consume;
310 }
311
312 /**
313 * Handles key events in a workspace containing icons.
314 */
315 static boolean handleIconKeyEvent(View v, int keyCode, KeyEvent e) {
316 boolean consume = FocusLogic.shouldConsume(keyCode);
317 if (e.getAction() == KeyEvent.ACTION_UP || !consume) {
318 return consume;
319 }
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700320
Adam Cohen2e6da152015-05-06 11:42:25 -0700321 Launcher launcher = (Launcher) v.getContext();
322 DeviceProfile profile = launcher.getDeviceProfile();
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700323
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800324 if (DEBUG) {
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700325 Log.v(TAG, String.format("Handle WORKSPACE ICONS keyevent=[%s] isVerticalBar=%s",
326 KeyEvent.keyCodeToString(keyCode), profile.isVerticalBarLayout()));
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800327 }
328
329 // Initialize the variables.
330 ShortcutAndWidgetContainer parent = (ShortcutAndWidgetContainer) v.getParent();
Hyunyoung Song38531712015-03-03 19:25:16 -0800331 CellLayout iconLayout = (CellLayout) parent.getParent();
Hyunyoung Song31178b82015-02-24 14:12:51 -0800332 final Workspace workspace = (Workspace) iconLayout.getParent();
Adam Cohen2e6da152015-05-06 11:42:25 -0700333 final ViewGroup dragLayer = (ViewGroup) workspace.getParent();
334 final ViewGroup tabs = (ViewGroup) dragLayer.findViewById(R.id.search_drop_target_bar);
335 final Hotseat hotseat = (Hotseat) dragLayer.findViewById(R.id.hotseat);
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700336
Winsonfa56b3f2015-09-14 12:01:13 -0700337 final ItemInfo itemInfo = (ItemInfo) v.getTag();
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700338 final int iconIndex = parent.indexOfChild(v);
339 final int pageIndex = workspace.indexOfChild(iconLayout);
340 final int pageCount = workspace.getChildCount();
Hyunyoung Song31178b82015-02-24 14:12:51 -0800341 int countX = iconLayout.getCountX();
342 int countY = iconLayout.getCountY();
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800343
344 CellLayout hotseatLayout = (CellLayout) hotseat.getChildAt(0);
345 ShortcutAndWidgetContainer hotseatParent = hotseatLayout.getShortcutsAndWidgets();
346 int[][] matrix;
347
Hyunyoung Song31178b82015-02-24 14:12:51 -0800348 // KEYCODE_DPAD_DOWN in portrait (KEYCODE_DPAD_RIGHT in landscape) is the only key allowed
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800349 // to take a user to the hotseat. For other dpad navigation, do not use the matrix extended
350 // with the hotseat.
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700351 if (keyCode == KeyEvent.KEYCODE_DPAD_DOWN && !profile.isVerticalBarLayout()) {
352 matrix = FocusLogic.createSparseMatrix(iconLayout, hotseatLayout, true /* horizontal */,
Tony Wickham6cbd2222015-11-09 17:51:08 -0800353 profile.inv.hotseatAllAppsRank);
354 countX = hotseatLayout.getCountX();
355 countY = countY + hotseatLayout.getCountY();
Hyunyoung Song31178b82015-02-24 14:12:51 -0800356 } else if (keyCode == KeyEvent.KEYCODE_DPAD_RIGHT &&
Hyunyoung Song18bfaaf2015-03-17 11:32:21 -0700357 profile.isVerticalBarLayout()) {
358 matrix = FocusLogic.createSparseMatrix(iconLayout, hotseatLayout, false /* horizontal */,
Tony Wickham6cbd2222015-11-09 17:51:08 -0800359 profile.inv.hotseatAllAppsRank);
360 countX = countX + hotseatLayout.getCountX();
361 countY = hotseatLayout.getCountY();
Winsonfa56b3f2015-09-14 12:01:13 -0700362 } else if (isUninstallKeyChord(e)) {
363 matrix = FocusLogic.createSparseMatrix(iconLayout);
364 if (UninstallDropTarget.supportsDrop(launcher, itemInfo)) {
365 UninstallDropTarget.startUninstallActivity(launcher, itemInfo);
366 }
367 } else if (isDeleteKeyChord(e)) {
368 matrix = FocusLogic.createSparseMatrix(iconLayout);
Winson2949fb52015-09-24 09:56:11 -0700369 launcher.removeItem(v, itemInfo, true /* deleteFromDb */);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800370 } else {
Hyunyoung Song31178b82015-02-24 14:12:51 -0800371 matrix = FocusLogic.createSparseMatrix(iconLayout);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800372 }
373
374 // Process the focus.
Adam Cohen2e6da152015-05-06 11:42:25 -0700375 int newIconIndex = FocusLogic.handleKeyEvent(keyCode, countX,
Sunny Goyalc6205602015-05-21 20:46:33 -0700376 countY, matrix, iconIndex, pageIndex, pageCount, Utilities.isRtl(v.getResources()));
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800377 View newIcon = null;
378 switch (newIconIndex) {
379 case FocusLogic.NOOP:
380 if (keyCode == KeyEvent.KEYCODE_DPAD_UP) {
381 newIcon = tabs;
382 }
383 break;
Hyunyoung Song38531712015-03-03 19:25:16 -0800384 case FocusLogic.PREVIOUS_PAGE_RIGHT_COLUMN:
Hyunyoung Songada50982015-04-10 14:35:23 -0700385 case FocusLogic.NEXT_PAGE_RIGHT_COLUMN:
386 int newPageIndex = pageIndex - 1;
387 if (newIconIndex == FocusLogic.NEXT_PAGE_RIGHT_COLUMN) {
388 newPageIndex = pageIndex + 1;
389 }
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700390 int row = ((CellLayout.LayoutParams) v.getLayoutParams()).cellY;
Hyunyoung Songada50982015-04-10 14:35:23 -0700391 parent = getCellLayoutChildrenForIndex(workspace, newPageIndex);
Hyunyoung Song38531712015-03-03 19:25:16 -0800392 if (parent != null) {
393 iconLayout = (CellLayout) parent.getParent();
Hyunyoung Songac721f82015-03-04 16:33:56 -0800394 matrix = FocusLogic.createSparseMatrix(iconLayout,
Tony Wickham4fc82872015-11-10 16:52:14 -0800395 iconLayout.getCountX(), row);
Adam Cohen2e6da152015-05-06 11:42:25 -0700396 newIconIndex = FocusLogic.handleKeyEvent(keyCode, countX + 1, countY,
Sunny Goyalc6205602015-05-21 20:46:33 -0700397 matrix, FocusLogic.PIVOT, newPageIndex, pageCount,
398 Utilities.isRtl(v.getResources()));
Hyunyoung Song38531712015-03-03 19:25:16 -0800399 newIcon = parent.getChildAt(newIconIndex);
400 }
401 break;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800402 case FocusLogic.PREVIOUS_PAGE_FIRST_ITEM:
403 parent = getCellLayoutChildrenForIndex(workspace, pageIndex - 1);
404 newIcon = parent.getChildAt(0);
Hyunyoung Song38531712015-03-03 19:25:16 -0800405 break;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800406 case FocusLogic.PREVIOUS_PAGE_LAST_ITEM:
407 parent = getCellLayoutChildrenForIndex(workspace, pageIndex - 1);
408 newIcon = parent.getChildAt(parent.getChildCount() - 1);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800409 break;
410 case FocusLogic.NEXT_PAGE_FIRST_ITEM:
411 parent = getCellLayoutChildrenForIndex(workspace, pageIndex + 1);
412 newIcon = parent.getChildAt(0);
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800413 break;
Hyunyoung Song38531712015-03-03 19:25:16 -0800414 case FocusLogic.NEXT_PAGE_LEFT_COLUMN:
Hyunyoung Songada50982015-04-10 14:35:23 -0700415 case FocusLogic.PREVIOUS_PAGE_LEFT_COLUMN:
416 newPageIndex = pageIndex + 1;
417 if (newIconIndex == FocusLogic.PREVIOUS_PAGE_LEFT_COLUMN) {
418 newPageIndex = pageIndex - 1;
419 }
Sunny Goyalfc3c1ed2015-04-09 18:48:21 -0700420 row = ((CellLayout.LayoutParams) v.getLayoutParams()).cellY;
Hyunyoung Songada50982015-04-10 14:35:23 -0700421 parent = getCellLayoutChildrenForIndex(workspace, newPageIndex);
Hyunyoung Song38531712015-03-03 19:25:16 -0800422 if (parent != null) {
423 iconLayout = (CellLayout) parent.getParent();
Hyunyoung Songac721f82015-03-04 16:33:56 -0800424 matrix = FocusLogic.createSparseMatrix(iconLayout, -1, row);
Adam Cohen2e6da152015-05-06 11:42:25 -0700425 newIconIndex = FocusLogic.handleKeyEvent(keyCode, countX + 1, countY,
Sunny Goyalc6205602015-05-21 20:46:33 -0700426 matrix, FocusLogic.PIVOT, newPageIndex, pageCount,
427 Utilities.isRtl(v.getResources()));
Hyunyoung Song38531712015-03-03 19:25:16 -0800428 newIcon = parent.getChildAt(newIconIndex);
429 }
430 break;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800431 case FocusLogic.CURRENT_PAGE_FIRST_ITEM:
432 newIcon = parent.getChildAt(0);
433 break;
434 case FocusLogic.CURRENT_PAGE_LAST_ITEM:
435 newIcon = parent.getChildAt(parent.getChildCount() - 1);
436 break;
437 default:
438 // current page, some item.
439 if (0 <= newIconIndex && newIconIndex < parent.getChildCount()) {
440 newIcon = parent.getChildAt(newIconIndex);
441 } else if (parent.getChildCount() <= newIconIndex &&
442 newIconIndex < parent.getChildCount() + hotseatParent.getChildCount()) {
443 newIcon = hotseatParent.getChildAt(newIconIndex - parent.getChildCount());
444 }
445 break;
446 }
447 if (newIcon != null) {
448 newIcon.requestFocus();
449 playSoundEffect(keyCode, v);
450 }
451 return consume;
452 }
453
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800454 //
455 // Helper methods.
456 //
457
Winson Chung97d85d22011-04-13 11:27:36 -0700458 /**
Winson Chung97d85d22011-04-13 11:27:36 -0700459 * Private helper method to get the CellLayoutChildren given a CellLayout index.
460 */
Sunny Goyal316490e2015-06-02 09:38:28 -0700461 @Thunk static ShortcutAndWidgetContainer getCellLayoutChildrenForIndex(
Michael Jurkaa52570f2012-03-20 03:18:20 -0700462 ViewGroup container, int i) {
Sunny Goyalb3726d92014-08-20 16:58:17 -0700463 CellLayout parent = (CellLayout) container.getChildAt(i);
464 return parent.getShortcutsAndWidgets();
Winson Chung97d85d22011-04-13 11:27:36 -0700465 }
466
Winson Chung97d85d22011-04-13 11:27:36 -0700467 /**
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800468 * Helper method to be used for playing sound effects.
Winson Chung97d85d22011-04-13 11:27:36 -0700469 */
Adam Cohen091440a2015-03-18 14:16:05 -0700470 @Thunk static void playSoundEffect(int keyCode, View v) {
Winson Chung97d85d22011-04-13 11:27:36 -0700471 switch (keyCode) {
472 case KeyEvent.KEYCODE_DPAD_LEFT:
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800473 v.playSoundEffect(SoundEffectConstants.NAVIGATION_LEFT);
Winson Chung97d85d22011-04-13 11:27:36 -0700474 break;
475 case KeyEvent.KEYCODE_DPAD_RIGHT:
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800476 v.playSoundEffect(SoundEffectConstants.NAVIGATION_RIGHT);
Winson Chung97d85d22011-04-13 11:27:36 -0700477 break;
478 case KeyEvent.KEYCODE_DPAD_DOWN:
Winson Chung97d85d22011-04-13 11:27:36 -0700479 case KeyEvent.KEYCODE_PAGE_DOWN:
Winson Chung97d85d22011-04-13 11:27:36 -0700480 case KeyEvent.KEYCODE_MOVE_END:
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800481 v.playSoundEffect(SoundEffectConstants.NAVIGATION_DOWN);
Adam Cohenac56cff2011-09-28 20:45:37 -0700482 break;
483 case KeyEvent.KEYCODE_DPAD_UP:
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800484 case KeyEvent.KEYCODE_PAGE_UP:
Adam Cohenac56cff2011-09-28 20:45:37 -0700485 case KeyEvent.KEYCODE_MOVE_HOME:
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800486 v.playSoundEffect(SoundEffectConstants.NAVIGATION_UP);
Adam Cohenac56cff2011-09-28 20:45:37 -0700487 break;
Hyunyoung Songee3e6a72015-02-20 14:25:27 -0800488 default:
Winson Chung97d85d22011-04-13 11:27:36 -0700489 break;
Winson Chung97d85d22011-04-13 11:27:36 -0700490 }
Winson Chung97d85d22011-04-13 11:27:36 -0700491 }
Winsonfa56b3f2015-09-14 12:01:13 -0700492
493 /**
494 * Returns whether the key event represents a valid uninstall key chord.
495 */
496 private static boolean isUninstallKeyChord(KeyEvent event) {
497 int keyCode = event.getKeyCode();
498 return (keyCode == KeyEvent.KEYCODE_DEL || keyCode == KeyEvent.KEYCODE_FORWARD_DEL) &&
499 event.hasModifiers(KeyEvent.META_CTRL_ON | KeyEvent.META_SHIFT_ON);
500 }
501
502 /**
503 * Returns whether the key event represents a valid delete key chord.
504 */
505 private static boolean isDeleteKeyChord(KeyEvent event) {
506 int keyCode = event.getKeyCode();
507 return (keyCode == KeyEvent.KEYCODE_DEL || keyCode == KeyEvent.KEYCODE_FORWARD_DEL) &&
508 event.hasModifiers(KeyEvent.META_CTRL_ON);
509 }
Winson Chung97d85d22011-04-13 11:27:36 -0700510}