19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/*
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License.
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License.
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpackage android.view;
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.Rect;
20140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mountimport android.util.ArrayMap;
21140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mountimport android.util.SparseArray;
22140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mountimport android.util.SparseBooleanArray;
239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.ArrayList;
254e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brownimport java.util.Collections;
264e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brownimport java.util.Comparator;
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/**
299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The algorithm used for finding the next focusable view in a given direction
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * from a view that currently has focus.
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpublic class FocusFinder {
339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3476f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov    private static final ThreadLocal<FocusFinder> tlFocusFinder =
359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            new ThreadLocal<FocusFinder>() {
364213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov                @Override
379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                protected FocusFinder initialValue() {
389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    return new FocusFinder();
399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            };
419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Get the focus finder for this thread.
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static FocusFinder getInstance() {
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return tlFocusFinder.get();
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4976f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov    final Rect mFocusedRect = new Rect();
5076f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov    final Rect mOtherRect = new Rect();
5176f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov    final Rect mBestCandidateRect = new Rect();
5276f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov    final SequentialFocusComparator mSequentialFocusComparator = new SequentialFocusComparator();
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
544213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov    private final ArrayList<View> mTempList = new ArrayList<View>();
554213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov
569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // enforce thread local access
579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private FocusFinder() {}
589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Find the next view to take focus in root's descendants, starting from the view
619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * that currently is focused.
627e0a372978eddf21808bf7fdfe36c1baa7f77e7cFabrice Di Meglio     * @param root Contains focused. Cannot be null.
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param focused Has focus now.
649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param direction Direction to look.
659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return The next focusable view, or null if none exists.
669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public final View findNextFocus(ViewGroup root, View focused, int direction) {
68951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov        return findNextFocus(root, focused, null, direction);
694213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov    }
709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
714213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov    /**
724213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov     * Find the next view to take focus in root's descendants, searching from
734213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov     * a particular rectangle in root's coordinates.
744213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov     * @param root Contains focusedRect. Cannot be null.
754213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov     * @param focusedRect The starting point of the search.
764213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov     * @param direction Direction to look.
774213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov     * @return The next focusable view, or null if none exists.
784213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov     */
794213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov    public View findNextFocusFromRect(ViewGroup root, Rect focusedRect, int direction) {
8076f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        mFocusedRect.set(focusedRect);
8176f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        return findNextFocus(root, null, mFocusedRect, direction);
824213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov    }
834213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov
844213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov    private View findNextFocus(ViewGroup root, View focused, Rect focusedRect, int direction) {
8576f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        View next = null;
869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (focused != null) {
8727e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov            next = findNextUserSpecifiedFocus(root, focused, direction);
8876f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        }
8976f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        if (next != null) {
9076f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            return next;
9176f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        }
9276f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        ArrayList<View> focusables = mTempList;
9376f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        try {
9476f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            focusables.clear();
9576f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            root.addFocusables(focusables, direction);
9676f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            if (!focusables.isEmpty()) {
9776f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov                next = findNextFocus(root, focused, focusedRect, direction, focusables);
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
9976f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        } finally {
10076f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            focusables.clear();
10176f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        }
10276f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        return next;
10376f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov    }
10476f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov
10527e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov    private View findNextUserSpecifiedFocus(ViewGroup root, View focused, int direction) {
10676f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        // check for user specified next focus
10776f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        View userSetNextFocus = focused.findUserSetNextFocus(root, direction);
10876f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        if (userSetNextFocus != null && userSetNextFocus.isFocusable()
10976f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov                && (!userSetNextFocus.isInTouchMode()
11076f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov                        || userSetNextFocus.isFocusableInTouchMode())) {
11176f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            return userSetNextFocus;
11276f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        }
11376f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        return null;
11476f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov    }
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
11676f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov    private View findNextFocus(ViewGroup root, View focused, Rect focusedRect,
11776f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            int direction, ArrayList<View> focusables) {
11876f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        if (focused != null) {
119951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov            if (focusedRect == null) {
120951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                focusedRect = mFocusedRect;
121951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov            }
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // fill in interesting rect from focused
12376f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            focused.getFocusedRect(focusedRect);
12476f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            root.offsetDescendantRectToMyCoords(focused, focusedRect);
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
126951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov            if (focusedRect == null) {
127951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                focusedRect = mFocusedRect;
128951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                // make up a rect at top left or bottom right of root
12927e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov                switch (direction) {
130951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                    case View.FOCUS_RIGHT:
131951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                    case View.FOCUS_DOWN:
13276f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov                        setFocusTopLeft(root, focusedRect);
133951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                        break;
134951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                    case View.FOCUS_FORWARD:
135951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                        if (root.isLayoutRtl()) {
136951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                            setFocusBottomRight(root, focusedRect);
137951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                        } else {
138951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                            setFocusTopLeft(root, focusedRect);
139951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                        }
140951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                        break;
141951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov
142951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                    case View.FOCUS_LEFT:
143951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                    case View.FOCUS_UP:
14476f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov                        setFocusBottomRight(root, focusedRect);
145951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                        break;
146951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                    case View.FOCUS_BACKWARD:
147951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                        if (root.isLayoutRtl()) {
148951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                            setFocusTopLeft(root, focusedRect);
149951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                        } else {
150951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                            setFocusBottomRight(root, focusedRect);
151951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                        break;
152951bb421668b82ca014f75d265b161f6ff64d36bSvetoslav Ganov                    }
153702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio                }
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
15727e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov        switch (direction) {
15876f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            case View.FOCUS_FORWARD:
15976f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            case View.FOCUS_BACKWARD:
16027e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov                return findNextFocusInRelativeDirection(focusables, root, focused, focusedRect,
16127e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov                        direction);
16276f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            case View.FOCUS_UP:
16376f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            case View.FOCUS_DOWN:
16476f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            case View.FOCUS_LEFT:
16576f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            case View.FOCUS_RIGHT:
16627e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov                return findNextFocusInAbsoluteDirection(focusables, root, focused,
16727e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov                        focusedRect, direction);
16876f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov            default:
16927e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov                throw new IllegalArgumentException("Unknown direction: " + direction);
1704213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        }
171702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio    }
172702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio
17327e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov    private View findNextFocusInRelativeDirection(ArrayList<View> focusables, ViewGroup root,
1744213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov            View focused, Rect focusedRect, int direction) {
1754213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        try {
1764213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov            // Note: This sort is stable.
1774213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov            mSequentialFocusComparator.setRoot(root);
1783167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio            mSequentialFocusComparator.setIsLayoutRtl(root.isLayoutRtl());
179140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            mSequentialFocusComparator.setFocusables(focusables);
1804213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov            Collections.sort(focusables, mSequentialFocusComparator);
1814213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        } finally {
1824213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov            mSequentialFocusComparator.recycle();
1834e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        }
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1854213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        final int count = focusables.size();
1864213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        switch (direction) {
1874213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov            case View.FOCUS_FORWARD:
1883167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio                return getNextFocusable(focused, focusables, count);
1894213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov            case View.FOCUS_BACKWARD:
1903167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio                return getPreviousFocusable(focused, focusables, count);
1914213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        }
1924213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        return focusables.get(count - 1);
1934213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov    }
1944213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov
19576f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov    private void setFocusBottomRight(ViewGroup root, Rect focusedRect) {
1964213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        final int rootBottom = root.getScrollY() + root.getHeight();
1974213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        final int rootRight = root.getScrollX() + root.getWidth();
19876f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        focusedRect.set(rootRight, rootBottom, rootRight, rootBottom);
1994213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov    }
2004213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov
20176f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov    private void setFocusTopLeft(ViewGroup root, Rect focusedRect) {
2024213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        final int rootTop = root.getScrollY();
2034213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov        final int rootLeft = root.getScrollX();
20476f287e416ded85734b610f316e38d243d2ddb09Svetoslav Ganov        focusedRect.set(rootLeft, rootTop, rootLeft, rootTop);
2054213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov    }
2064213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov
20727e2da7c171afa39358bbead18fbe3e6b8ea6637Svetoslav Ganov    View findNextFocusInAbsoluteDirection(ArrayList<View> focusables, ViewGroup root, View focused,
2084213804541a8b05cd0587b138a2fd9a3b7fd9350Svetoslav Ganov            Rect focusedRect, int direction) {
2099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // initialize the best candidate to something impossible
2109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // (so the first plausible view will become the best choice)
2119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mBestCandidateRect.set(focusedRect);
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch(direction) {
2139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_LEFT:
2149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mBestCandidateRect.offset(focusedRect.width() + 1, 0);
2159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
2169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_RIGHT:
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mBestCandidateRect.offset(-(focusedRect.width() + 1), 0);
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
2199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_UP:
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mBestCandidateRect.offset(0, focusedRect.height() + 1);
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
2229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_DOWN:
2239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mBestCandidateRect.offset(0, -(focusedRect.height() + 1));
2249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        View closest = null;
2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int numFocusables = focusables.size();
2299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (int i = 0; i < numFocusables; i++) {
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            View focusable = focusables.get(i);
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // only interested in other non-root views
2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (focusable == focused || focusable == root) continue;
2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
235defdb1e49172fe7c9737347489dbb77361af955aTobias Dubois            // get focus bounds of other view in same coordinate system
236c11f77fbae8391ca3c2d3ec93d024cba0be5cf55Fabrice Di Meglio            focusable.getFocusedRect(mOtherRect);
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            root.offsetDescendantRectToMyCoords(focusable, mOtherRect);
2389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (isBetterCandidate(direction, focusedRect, mOtherRect, mBestCandidateRect)) {
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mBestCandidateRect.set(mOtherRect);
2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                closest = focusable;
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return closest;
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2477e0a372978eddf21808bf7fdfe36c1baa7f77e7cFabrice Di Meglio    private static View getNextFocusable(View focused, ArrayList<View> focusables, int count) {
248702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio        if (focused != null) {
249702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio            int position = focusables.lastIndexOf(focused);
250702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio            if (position >= 0 && position + 1 < count) {
251702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio                return focusables.get(position + 1);
252702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio            }
253702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio        }
254e5dfa47d84668376b84074c04570fb961870adebSvetoslav Ganov        if (!focusables.isEmpty()) {
255e5dfa47d84668376b84074c04570fb961870adebSvetoslav Ganov            return focusables.get(0);
256e5dfa47d84668376b84074c04570fb961870adebSvetoslav Ganov        }
257e5dfa47d84668376b84074c04570fb961870adebSvetoslav Ganov        return null;
258702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio    }
259702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio
2607e0a372978eddf21808bf7fdfe36c1baa7f77e7cFabrice Di Meglio    private static View getPreviousFocusable(View focused, ArrayList<View> focusables, int count) {
261702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio        if (focused != null) {
262702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio            int position = focusables.indexOf(focused);
263702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio            if (position > 0) {
264702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio                return focusables.get(position - 1);
265702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio            }
266702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio        }
267e5dfa47d84668376b84074c04570fb961870adebSvetoslav Ganov        if (!focusables.isEmpty()) {
268e5dfa47d84668376b84074c04570fb961870adebSvetoslav Ganov            return focusables.get(count - 1);
269e5dfa47d84668376b84074c04570fb961870adebSvetoslav Ganov        }
270e5dfa47d84668376b84074c04570fb961870adebSvetoslav Ganov        return null;
271702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio    }
272702e8f9b9294d8227deae6e1f125a768ee4a28d4Fabrice Di Meglio
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Is rect1 a better candidate than rect2 for a focus search in a particular
2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * direction from a source rect?  This is the core routine that determines
2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the order of focus searching.
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param direction the direction (up, down, left, right)
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param source The source we are searching from
2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param rect1 The candidate rectangle
2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param rect2 The current best candidate.
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return Whether the candidate is the new best.
2829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    boolean isBetterCandidate(int direction, Rect source, Rect rect1, Rect rect2) {
2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // to be a better candidate, need to at least be a candidate in the first
2869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // place :)
2879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!isCandidate(source, rect1, direction)) {
2889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return false;
2899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // we know that rect1 is a candidate.. if rect2 is not a candidate,
2929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // rect1 is better
2939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!isCandidate(source, rect2, direction)) {
2949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return true;
2959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // if rect1 is better by beam, it wins
2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (beamBeats(direction, source, rect1, rect2)) {
2999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return true;
3009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // if rect2 is better, then rect1 cant' be :)
3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (beamBeats(direction, source, rect2, rect1)) {
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return false;
3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // otherwise, do fudge-tastic comparison of the major and minor axis
3089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return (getWeightedDistanceFor(
3099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        majorAxisDistance(direction, source, rect1),
3109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        minorAxisDistance(direction, source, rect1))
3119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                < getWeightedDistanceFor(
3129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        majorAxisDistance(direction, source, rect2),
3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        minorAxisDistance(direction, source, rect2)));
3149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * One rectangle may be another candidate than another by virtue of being
3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * exclusively in the beam of the source rect.
3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return Whether rect1 is a better candidate than rect2 by virtue of it being in src's
3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *      beam
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    boolean beamBeats(int direction, Rect source, Rect rect1, Rect rect2) {
3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final boolean rect1InSrcBeam = beamsOverlap(direction, source, rect1);
3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final boolean rect2InSrcBeam = beamsOverlap(direction, source, rect2);
3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // if rect1 isn't exclusively in the src beam, it doesn't win
3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (rect2InSrcBeam || !rect1InSrcBeam) {
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return false;
3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // we know rect1 is in the beam, and rect2 is not
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // if rect1 is to the direction of, and rect2 is not, rect1 wins.
3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // for example, for direction left, if rect1 is to the left of the source
3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // and rect2 is below, then we always prefer the in beam rect1, since rect2
3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // could be reached by going down.
3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (!isToDirectionOf(direction, source, rect2)) {
3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return true;
3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // for horizontal directions, being exclusively in beam always wins
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if ((direction == View.FOCUS_LEFT || direction == View.FOCUS_RIGHT)) {
3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return true;
3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // for vertical directions, beams only beat up to a point:
3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // now, as long as rect2 isn't completely closer, rect1 wins
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // e.g for direction down, completely closer means for rect2's top
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // edge to be closer to the source's top edge than rect1's bottom edge.
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return (majorAxisDistance(direction, source, rect1)
3519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                < majorAxisDistanceToFarEdge(direction, source, rect2));
3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Fudge-factor opportunity: how to calculate distance given major and minor
3569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * axis distances.  Warning: this fudge factor is finely tuned, be sure to
3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * run all focus tests if you dare tweak it.
3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int getWeightedDistanceFor(int majorAxisDistance, int minorAxisDistance) {
3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return 13 * majorAxisDistance * majorAxisDistance
3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + minorAxisDistance * minorAxisDistance;
3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Is destRect a candidate for the next focus given the direction?  This
3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * checks whether the dest is at least partially to the direction of (e.g left of)
3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * from source.
3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Includes an edge case for an empty rect (which is used in some cases when
3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * searching from a point on the screen).
3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    boolean isCandidate(Rect srcRect, Rect destRect, int direction) {
3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (direction) {
3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_LEFT:
3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return (srcRect.right > destRect.right || srcRect.left >= destRect.right)
3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        && srcRect.left > destRect.left;
3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_RIGHT:
3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return (srcRect.left < destRect.left || srcRect.right <= destRect.left)
3799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        && srcRect.right < destRect.right;
3809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_UP:
3819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return (srcRect.bottom > destRect.bottom || srcRect.top >= destRect.bottom)
3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        && srcRect.top > destRect.top;
3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_DOWN:
3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return (srcRect.top < destRect.top || srcRect.bottom <= destRect.top)
3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        && srcRect.bottom < destRect.bottom;
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        throw new IllegalArgumentException("direction must be one of "
3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + "{FOCUS_UP, FOCUS_DOWN, FOCUS_LEFT, FOCUS_RIGHT}.");
3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
3937e0a372978eddf21808bf7fdfe36c1baa7f77e7cFabrice Di Meglio     * Do the "beams" w.r.t the given direction's axis of rect1 and rect2 overlap?
3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param direction the direction (up, down, left, right)
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param rect1 The first rectangle
3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param rect2 The second rectangle
3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return whether the beams overlap
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    boolean beamsOverlap(int direction, Rect rect1, Rect rect2) {
4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (direction) {
4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_LEFT:
4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_RIGHT:
4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return (rect2.bottom >= rect1.top) && (rect2.top <= rect1.bottom);
4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_UP:
4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_DOWN:
4069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return (rect2.right >= rect1.left) && (rect2.left <= rect1.right);
4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        throw new IllegalArgumentException("direction must be one of "
4099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + "{FOCUS_UP, FOCUS_DOWN, FOCUS_LEFT, FOCUS_RIGHT}.");
4109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * e.g for left, is 'to left of'
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    boolean isToDirectionOf(int direction, Rect src, Rect dest) {
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (direction) {
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_LEFT:
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return src.left >= dest.right;
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_RIGHT:
4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return src.right <= dest.left;
4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_UP:
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return src.top >= dest.bottom;
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_DOWN:
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return src.bottom <= dest.top;
4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        throw new IllegalArgumentException("direction must be one of "
4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + "{FOCUS_UP, FOCUS_DOWN, FOCUS_LEFT, FOCUS_RIGHT}.");
4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return The distance from the edge furthest in the given direction
4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *   of source to the edge nearest in the given direction of dest.  If the
4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *   dest is not in the direction from source, return 0.
4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int majorAxisDistance(int direction, Rect source, Rect dest) {
4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return Math.max(0, majorAxisDistanceRaw(direction, source, dest));
4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int majorAxisDistanceRaw(int direction, Rect source, Rect dest) {
4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (direction) {
4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_LEFT:
4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return source.left - dest.right;
4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_RIGHT:
4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return dest.left - source.right;
4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_UP:
4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return source.top - dest.bottom;
4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_DOWN:
4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return dest.top - source.bottom;
4499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        throw new IllegalArgumentException("direction must be one of "
4519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + "{FOCUS_UP, FOCUS_DOWN, FOCUS_LEFT, FOCUS_RIGHT}.");
4529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
4559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return The distance along the major axis w.r.t the direction from the
4569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *   edge of source to the far edge of dest. If the
4579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *   dest is not in the direction from source, return 1 (to break ties with
4589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *   {@link #majorAxisDistance}).
4599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int majorAxisDistanceToFarEdge(int direction, Rect source, Rect dest) {
4619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return Math.max(1, majorAxisDistanceToFarEdgeRaw(direction, source, dest));
4629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int majorAxisDistanceToFarEdgeRaw(int direction, Rect source, Rect dest) {
4659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (direction) {
4669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_LEFT:
4679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return source.left - dest.left;
4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_RIGHT:
4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return dest.right - source.right;
4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_UP:
4719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return source.top - dest.top;
4729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_DOWN:
4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return dest.bottom - source.bottom;
4749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        throw new IllegalArgumentException("direction must be one of "
4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + "{FOCUS_UP, FOCUS_DOWN, FOCUS_LEFT, FOCUS_RIGHT}.");
4779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Find the distance on the minor axis w.r.t the direction to the nearest
4817e0a372978eddf21808bf7fdfe36c1baa7f77e7cFabrice Di Meglio     * edge of the destination rectangle.
4829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param direction the direction (up, down, left, right)
4839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param source The source rect.
4849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param dest The destination rect.
4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return The distance.
4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static int minorAxisDistance(int direction, Rect source, Rect dest) {
4889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (direction) {
4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_LEFT:
4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_RIGHT:
4919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // the distance between the center verticals
4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return Math.abs(
4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        ((source.top + source.height() / 2) -
4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        ((dest.top + dest.height() / 2))));
4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_UP:
4969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_DOWN:
4979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // the distance between the center horizontals
4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return Math.abs(
4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        ((source.left + source.width() / 2) -
5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        ((dest.left + dest.width() / 2))));
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        throw new IllegalArgumentException("direction must be one of "
5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + "{FOCUS_UP, FOCUS_DOWN, FOCUS_LEFT, FOCUS_RIGHT}.");
5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Find the nearest touchable view to the specified view.
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param root The root of the tree in which to search
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param x X coordinate from which to start the search
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param y Y coordinate from which to start the search
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param direction Direction to look
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param deltas Offset from the <x, y> to the edge of the nearest view. Note that this array
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *        may already be populated with values.
5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return The nearest touchable view, or null if none exists.
5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public View findNearestTouchable(ViewGroup root, int x, int y, int direction, int[] deltas) {
5189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        ArrayList<View> touchables = root.getTouchables();
5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int minDistance = Integer.MAX_VALUE;
5209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        View closest = null;
5219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int numTouchables = touchables.size();
5239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int edgeSlop = ViewConfiguration.get(root.mContext).getScaledEdgeSlop();
5259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Rect closestBounds = new Rect();
5279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        Rect touchableBounds = mOtherRect;
5289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (int i = 0; i < numTouchables; i++) {
5309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            View touchable = touchables.get(i);
5319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // get visible bounds of other view in same coordinate system
5339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            touchable.getDrawingRect(touchableBounds);
5349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            root.offsetRectBetweenParentAndChild(touchable, touchableBounds, true, true);
5369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (!isTouchCandidate(x, y, touchableBounds, direction)) {
5389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                continue;
5399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int distance = Integer.MAX_VALUE;
5429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            switch (direction) {
5449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_LEFT:
5459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                distance = x - touchableBounds.right + 1;
5469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
5479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_RIGHT:
5489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                distance = touchableBounds.left;
5499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
5509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_UP:
5519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                distance = y - touchableBounds.bottom + 1;
5529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
5539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_DOWN:
5549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                distance = touchableBounds.top;
5559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                break;
5569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (distance < edgeSlop) {
5599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Give preference to innermost views
5609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (closest == null ||
5619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        closestBounds.contains(touchableBounds) ||
5629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        (!touchableBounds.contains(closestBounds) && distance < minDistance)) {
5639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    minDistance = distance;
5649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    closest = touchable;
5659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    closestBounds.set(touchableBounds);
5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    switch (direction) {
5679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case View.FOCUS_LEFT:
5689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        deltas[0] = -distance;
5699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case View.FOCUS_RIGHT:
5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        deltas[0] = distance;
5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case View.FOCUS_UP:
5749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        deltas[1] = -distance;
5759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
5769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case View.FOCUS_DOWN:
5779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        deltas[1] = distance;
5789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
5799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
5809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
5819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return closest;
5849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
5889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Is destRect a candidate for the next touch given the direction?
5899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
5909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean isTouchCandidate(int x, int y, Rect destRect, int direction) {
5919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        switch (direction) {
5929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_LEFT:
5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return destRect.left <= x && destRect.top <= y && y <= destRect.bottom;
5949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_RIGHT:
5959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return destRect.left >= x && destRect.top <= y && y <= destRect.bottom;
5969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_UP:
5979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return destRect.top <= y && destRect.left <= x && x <= destRect.right;
5989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            case View.FOCUS_DOWN:
5999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return destRect.top >= y && destRect.left <= x && x <= destRect.right;
6009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        throw new IllegalArgumentException("direction must be one of "
6029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                + "{FOCUS_UP, FOCUS_DOWN, FOCUS_LEFT, FOCUS_RIGHT}.");
6039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6044e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown
605140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount    private static final boolean isValidId(final int id) {
606140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount        return id != 0 && id != View.NO_ID;
607140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount    }
608140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount
6094e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown    /**
6104e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown     * Sorts views according to their visual layout and geometry for default tab order.
611140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount     * If views are part of a focus chain (nextFocusForwardId), then they are all grouped
612140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount     * together. The head of the chain is used to determine the order of the chain and is
613140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount     * first in the order and the tail of the chain is the last in the order. The views
614140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount     * in the middle of the chain can be arbitrary order.
6154e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown     * This is used for sequential focus traversal.
6164e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown     */
6174e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown    private static final class SequentialFocusComparator implements Comparator<View> {
6184e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        private final Rect mFirstRect = new Rect();
6194e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        private final Rect mSecondRect = new Rect();
6204e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        private ViewGroup mRoot;
6213167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio        private boolean mIsLayoutRtl;
622140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount        private final SparseArray<View> mFocusables = new SparseArray<View>();
623140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount        private final SparseBooleanArray mIsConnectedTo = new SparseBooleanArray();
624140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount        private final ArrayMap<View, View> mHeadsOfChains = new ArrayMap<View, View>();
6254e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown
6264e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        public void recycle() {
6274e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            mRoot = null;
628140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            mFocusables.clear();
629140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            mHeadsOfChains.clear();
630140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            mIsConnectedTo.clear();
6314e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        }
6324e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown
6334e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        public void setRoot(ViewGroup root) {
6344e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            mRoot = root;
6354e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        }
6364e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown
6373167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio        public void setIsLayoutRtl(boolean b) {
6383167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio            mIsLayoutRtl = b;
6393167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio        }
6403167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio
641140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount        public void setFocusables(ArrayList<View> focusables) {
642140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            for (int i = focusables.size() - 1; i >= 0; i--) {
643140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                final View view = focusables.get(i);
644140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                final int id = view.getId();
645140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                if (isValidId(id)) {
646140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    mFocusables.put(id, view);
647140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                }
648140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                final int nextId = view.getNextFocusForwardId();
649140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                if (isValidId(nextId)) {
650140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    mIsConnectedTo.put(nextId, true);
651140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                }
652140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            }
653140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount
654140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            for (int i = focusables.size() - 1; i >= 0; i--) {
655140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                final View view = focusables.get(i);
656140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                final int nextId = view.getNextFocusForwardId();
657140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                if (isValidId(nextId) && !mIsConnectedTo.get(view.getId())) {
658140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    setHeadOfChain(view);
659140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                }
660140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            }
661140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount        }
662140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount
663140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount        private void setHeadOfChain(View head) {
664140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            for (View view = head; view != null;
665140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    view = mFocusables.get(view.getNextFocusForwardId())) {
666140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                final View otherHead = mHeadsOfChains.get(view);
667140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                if (otherHead != null) {
668140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    if (otherHead == head) {
669140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                        return; // This view has already had its head set properly
670140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    }
671140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    // A hydra -- multi-headed focus chain (e.g. A->C and B->C)
672140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    // Use the one we've already chosen instead and reset this chain.
673140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    view = head;
674140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    head = otherHead;
675140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                }
676140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                mHeadsOfChains.put(view, head);
677140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            }
678140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount        }
679140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount
6804e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        public int compare(View first, View second) {
6814e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            if (first == second) {
6824e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown                return 0;
6834e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            }
684140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            // Order between views within a chain is immaterial -- next/previous is
685140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            // within a chain is handled elsewhere.
686140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            View firstHead = mHeadsOfChains.get(first);
687140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            View secondHead = mHeadsOfChains.get(second);
688140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            if (firstHead == secondHead && firstHead != null) {
689140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                if (first == firstHead) {
690140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    return -1; // first is the head, it should be first
691140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                } else if (second == firstHead) {
692140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    return 1; // second is the head, it should be first
693140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                } else if (isValidId(first.getNextFocusForwardId())) {
694140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    return -1; // first is not the end of the chain
695140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                } else {
696140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                    return 1; // first is end of chain
697140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                }
698140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            }
699140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            if (firstHead != null) {
700140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                first = firstHead;
701140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            }
702140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            if (secondHead != null) {
703140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount                second = secondHead;
704140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            }
7054e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown
706140ea62c385cfe0b9ab28b8c2b78d0f61a118adaGeorge Mount            // First see if they belong to the same focus chain.
7074e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            getRect(first, mFirstRect);
7084e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            getRect(second, mSecondRect);
7094e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown
7104e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            if (mFirstRect.top < mSecondRect.top) {
7114e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown                return -1;
7124e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            } else if (mFirstRect.top > mSecondRect.top) {
7134e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown                return 1;
7144e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            } else if (mFirstRect.left < mSecondRect.left) {
7153167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio                return mIsLayoutRtl ? 1 : -1;
7164e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            } else if (mFirstRect.left > mSecondRect.left) {
7173167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio                return mIsLayoutRtl ? -1 : 1;
7184e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            } else if (mFirstRect.bottom < mSecondRect.bottom) {
7194e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown                return -1;
7204e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            } else if (mFirstRect.bottom > mSecondRect.bottom) {
7214e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown                return 1;
7224e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            } else if (mFirstRect.right < mSecondRect.right) {
7233167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio                return mIsLayoutRtl ? 1 : -1;
7244e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            } else if (mFirstRect.right > mSecondRect.right) {
7253167c88c2c18eaadb046d41e2108bf45eae0d289Fabrice Di Meglio                return mIsLayoutRtl ? -1 : 1;
7264e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            } else {
7274e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown                // The view are distinct but completely coincident so we consider
7284e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown                // them equal for our purposes.  Since the sort is stable, this
7294e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown                // means that the views will retain their layout order relative to one another.
7304e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown                return 0;
7314e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            }
7324e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        }
7334e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown
7344e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        private void getRect(View view, Rect rect) {
7354e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            view.getDrawingRect(rect);
7364e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown            mRoot.offsetDescendantRectToMyCoords(view, rect);
7374e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown        }
7384e6319b73c85082e18d1c532b86336ddd1f8cfaaJeff Brown    }
7399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
740