| /* |
| * Copyright (C) 2018 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License |
| */ |
| |
| package com.android.systemui.statusbar.phone; |
| |
| import static android.view.WindowManagerPolicyConstants.NAV_BAR_BOTTOM; |
| import static android.view.WindowManagerPolicyConstants.NAV_BAR_INVALID; |
| import static android.view.WindowManagerPolicyConstants.NAV_BAR_LEFT; |
| import static android.view.WindowManagerPolicyConstants.NAV_BAR_RIGHT; |
| |
| import static com.android.systemui.recents.OverviewProxyService.DEBUG_OVERVIEW_PROXY; |
| import static com.android.systemui.recents.OverviewProxyService.TAG_OPS; |
| import static com.android.systemui.shared.system.NavigationBarCompat.HIT_TARGET_BACK; |
| import static com.android.systemui.shared.system.NavigationBarCompat.HIT_TARGET_DEAD_ZONE; |
| import static com.android.systemui.shared.system.NavigationBarCompat.HIT_TARGET_HOME; |
| import static com.android.systemui.shared.system.NavigationBarCompat.HIT_TARGET_NONE; |
| import static com.android.systemui.shared.system.NavigationBarCompat.HIT_TARGET_OVERVIEW; |
| import static com.android.systemui.statusbar.phone.NavigationBarView.WINDOW_TARGET_BOTTOM; |
| |
| import android.annotation.Nullable; |
| import android.content.Context; |
| import android.content.res.Resources; |
| import android.graphics.Canvas; |
| import android.graphics.Matrix; |
| import android.graphics.Rect; |
| import android.hardware.input.InputManager; |
| import android.os.RemoteException; |
| import android.os.SystemClock; |
| import android.provider.Settings; |
| import android.util.Log; |
| import android.view.InputDevice; |
| import android.view.MotionEvent; |
| import android.view.View; |
| import android.view.ViewConfiguration; |
| import android.view.ViewPropertyAnimator; |
| |
| import com.android.systemui.Dependency; |
| import com.android.systemui.Interpolators; |
| import com.android.systemui.R; |
| import com.android.systemui.SysUiServiceProvider; |
| import com.android.systemui.plugins.statusbar.phone.NavGesture.GestureHelper; |
| import com.android.systemui.recents.OverviewProxyService; |
| import com.android.systemui.shared.recents.IOverviewProxy; |
| import com.android.systemui.shared.recents.utilities.Utilities; |
| import com.android.systemui.shared.system.InputChannelCompat.InputEventDispatcher; |
| import com.android.systemui.shared.system.NavigationBarCompat; |
| import com.android.systemui.shared.system.QuickStepContract; |
| |
| import java.io.PrintWriter; |
| |
| /** |
| * Class to detect gestures on the navigation bar and implement quick scrub. |
| * Note that the variables in this class horizontal and vertical represents horizontal always |
| * aligned with along the navigation bar. |
| */ |
| public class QuickStepController implements GestureHelper { |
| |
| private static final String TAG = "QuickStepController"; |
| |
| /** Experiment to swipe home button left to execute a back key press */ |
| private static final String HIDE_BACK_BUTTON_PROP = "quickstepcontroller_hideback"; |
| private static final String HIDE_HOME_BUTTON_PROP = "quickstepcontroller_hidehome"; |
| private static final String ENABLE_CLICK_THROUGH_NAV_PROP = "quickstepcontroller_clickthrough"; |
| private static final String GESTURE_REGION_THRESHOLD_SETTING = "gesture_region_threshold"; |
| private static final long BACK_BUTTON_FADE_IN_ALPHA = 150; |
| private static final long CLICK_THROUGH_TAP_DELAY = 70; |
| private static final long CLICK_THROUGH_TAP_RESET_DELAY = 100; |
| |
| /** When the home-swipe-back gesture is disallowed, make it harder to pull */ |
| private static final float HORIZONTAL_GESTURE_DAMPING = 0.3f; |
| private static final float VERTICAL_GESTURE_DAMPING = 0.15f; |
| private static final float HORIZONTAL_DISABLED_GESTURE_DAMPING = 0.16f; |
| private static final float VERTICAL_DISABLED_GESTURE_DAMPING = 0.06f; |
| |
| private static final int ACTION_SWIPE_UP_INDEX = 0; |
| private static final int ACTION_SWIPE_DOWN_INDEX = 1; |
| private static final int ACTION_SWIPE_LEFT_INDEX = 2; |
| private static final int ACTION_SWIPE_RIGHT_INDEX = 3; |
| private static final int ACTION_SWIPE_LEFT_FROM_EDGE_INDEX = 4; |
| private static final int ACTION_SWIPE_RIGHT_FROM_EDGE_INDEX = 5; |
| private static final int MAX_GESTURES = 6; |
| |
| private NavigationBarView mNavigationBarView; |
| |
| private boolean mAllowGestureDetection; |
| private boolean mNotificationsVisibleOnDown; |
| private int mTouchDownX; |
| private int mTouchDownY; |
| private boolean mDragHPositive; |
| private boolean mDragVPositive; |
| private boolean mIsRTL; |
| private int mNavBarPosition; |
| private float mDarkIntensity; |
| private ViewPropertyAnimator mDragBtnAnimator; |
| private ButtonDispatcher mHitTarget; |
| private boolean mIsInScreenPinning; |
| private boolean mGestureHorizontalDragsButton; |
| private boolean mGestureVerticalDragsButton; |
| private float mMaxDragLimit; |
| private float mMinDragLimit; |
| private float mDragDampeningFactor; |
| private boolean mClickThroughPressed; |
| private float mClickThroughPressX; |
| private float mClickThroughPressY; |
| private int mGestureRegionThreshold; |
| |
| private NavigationGestureAction mCurrentAction; |
| private NavigationGestureAction[] mGestureActions = new NavigationGestureAction[MAX_GESTURES]; |
| |
| private final Rect mLastLayoutRect = new Rect(); |
| private final OverviewProxyService mOverviewEventSender; |
| private final Context mContext; |
| private final StatusBar mStatusBar; |
| private final Matrix mTransformGlobalMatrix = new Matrix(); |
| private final Matrix mTransformLocalMatrix = new Matrix(); |
| |
| public QuickStepController(Context context) { |
| mContext = context; |
| mStatusBar = SysUiServiceProvider.getComponent(context, StatusBar.class); |
| mOverviewEventSender = Dependency.get(OverviewProxyService.class); |
| } |
| |
| private final Runnable mClickThroughSendTap = new Runnable() { |
| @Override |
| public void run() { |
| sendTap(mClickThroughPressX, mClickThroughPressY); |
| mNavigationBarView.postDelayed(mClickThroughResetTap, CLICK_THROUGH_TAP_RESET_DELAY); |
| } |
| }; |
| |
| private final Runnable mClickThroughResetTap = () -> { |
| mNavigationBarView.setWindowTouchable(true); |
| mClickThroughPressed = false; |
| }; |
| |
| public void setComponents(NavigationBarView navigationBarView) { |
| mNavigationBarView = navigationBarView; |
| |
| mNavigationBarView.getBackButton().setVisibility(shouldHideBackButton(mContext) |
| ? View.GONE |
| : View.VISIBLE); |
| } |
| |
| /** |
| * Set each gesture an action. After set the gestures triggered will run the actions attached. |
| * @param swipeUpAction action after swiping up |
| * @param swipeDownAction action after swiping down |
| * @param swipeLeftAction action after swiping left |
| * @param swipeRightAction action after swiping right |
| * @param swipeLeftFromEdgeAction action swiping left starting from the right side |
| * @param swipeRightFromEdgeAction action swiping right starting from the left side |
| */ |
| public void setGestureActions(@Nullable NavigationGestureAction swipeUpAction, |
| @Nullable NavigationGestureAction swipeDownAction, |
| @Nullable NavigationGestureAction swipeLeftAction, |
| @Nullable NavigationGestureAction swipeRightAction, |
| @Nullable NavigationGestureAction swipeLeftFromEdgeAction, |
| @Nullable NavigationGestureAction swipeRightFromEdgeAction) { |
| mGestureActions[ACTION_SWIPE_UP_INDEX] = swipeUpAction; |
| mGestureActions[ACTION_SWIPE_DOWN_INDEX] = swipeDownAction; |
| mGestureActions[ACTION_SWIPE_LEFT_INDEX] = swipeLeftAction; |
| mGestureActions[ACTION_SWIPE_RIGHT_INDEX] = swipeRightAction; |
| mGestureActions[ACTION_SWIPE_LEFT_FROM_EDGE_INDEX] = swipeLeftFromEdgeAction; |
| mGestureActions[ACTION_SWIPE_RIGHT_FROM_EDGE_INDEX] = swipeRightFromEdgeAction; |
| |
| // Set the current state to all actions |
| for (NavigationGestureAction action: mGestureActions) { |
| if (action != null) { |
| action.setBarState(true, mNavBarPosition, mDragHPositive, mDragVPositive); |
| action.onDarkIntensityChange(mDarkIntensity); |
| action.onLayout(true /* changed */, mLastLayoutRect.left, mLastLayoutRect.top, |
| mLastLayoutRect.right, mLastLayoutRect.bottom); |
| } |
| } |
| } |
| |
| /** |
| * @return true if we want to intercept touch events for quick scrub and prevent proxying the |
| * event to the overview service. |
| */ |
| @Override |
| public boolean onInterceptTouchEvent(MotionEvent event) { |
| return handleTouchEvent(event); |
| } |
| |
| /** |
| * @return true if we want to handle touch events for quick scrub or if down event (that will |
| * get consumed and ignored). No events will be proxied to the overview service. |
| */ |
| @Override |
| public boolean onTouchEvent(MotionEvent event) { |
| // The same down event was just sent on intercept and therefore can be ignored here |
| final boolean ignoreProxyDownEvent = event.getAction() == MotionEvent.ACTION_DOWN |
| && mOverviewEventSender.getProxy() != null |
| && mNavigationBarView.getWindowTarget() == WINDOW_TARGET_BOTTOM; |
| return ignoreProxyDownEvent || handleTouchEvent(event); |
| } |
| |
| private boolean handleTouchEvent(MotionEvent event) { |
| final boolean deadZoneConsumed = |
| mNavigationBarView.getDownHitTarget() == HIT_TARGET_DEAD_ZONE; |
| |
| // Requires proxy and an active gesture or able to perform any gesture to continue |
| if (mOverviewEventSender.getProxy() == null |
| || !mOverviewEventSender.shouldShowSwipeUpUI() |
| || (mCurrentAction == null && !canPerformAnyAction())) { |
| return deadZoneConsumed; |
| } |
| mNavigationBarView.requestUnbufferedDispatch(event); |
| |
| int action = event.getActionMasked(); |
| switch (action) { |
| case MotionEvent.ACTION_DOWN: { |
| int x = (int) event.getX(); |
| int y = (int) event.getY(); |
| mIsInScreenPinning = mNavigationBarView.inScreenPinning(); |
| |
| for (NavigationGestureAction gestureAction: mGestureActions) { |
| if (gestureAction != null) { |
| gestureAction.reset(); |
| } |
| } |
| |
| // Valid buttons to drag over |
| switch (mNavigationBarView.getDownHitTarget()) { |
| case HIT_TARGET_BACK: |
| mHitTarget = mNavigationBarView.getBackButton(); |
| break; |
| case HIT_TARGET_HOME: |
| mHitTarget = mNavigationBarView.getHomeButton(); |
| break; |
| case HIT_TARGET_OVERVIEW: |
| mHitTarget = mNavigationBarView.getRecentsButton(); |
| break; |
| default: |
| mHitTarget = null; |
| break; |
| } |
| if (mHitTarget != null) { |
| // Pre-emptively delay the touch feedback for the button that we just touched |
| mHitTarget.setDelayTouchFeedback(true); |
| } |
| mTouchDownX = x; |
| mTouchDownY = y; |
| mGestureHorizontalDragsButton = false; |
| mGestureVerticalDragsButton = false; |
| mTransformGlobalMatrix.set(Matrix.IDENTITY_MATRIX); |
| mTransformLocalMatrix.set(Matrix.IDENTITY_MATRIX); |
| mNavigationBarView.transformMatrixToGlobal(mTransformGlobalMatrix); |
| mNavigationBarView.transformMatrixToLocal(mTransformLocalMatrix); |
| mAllowGestureDetection = true; |
| mNotificationsVisibleOnDown = !mNavigationBarView.isNotificationsFullyCollapsed(); |
| mGestureRegionThreshold = QuickStepContract.getEdgeSensitivityWidth(mContext); |
| break; |
| } |
| case MotionEvent.ACTION_MOVE: { |
| if (!mAllowGestureDetection |
| || mNavigationBarView.getWindowTarget() != WINDOW_TARGET_BOTTOM) { |
| break; |
| } |
| int x = (int) event.getX(); |
| int y = (int) event.getY(); |
| int xDiff = Math.abs(x - mTouchDownX); |
| int yDiff = Math.abs(y - mTouchDownY); |
| |
| boolean exceededSwipeHorizontalTouchSlop, exceededSwipeVerticalTouchSlop, |
| exceededSwipeVerticalDragSlop; |
| int posH, touchDownH, posV, touchDownV; |
| |
| if (isNavBarVertical()) { |
| exceededSwipeHorizontalTouchSlop = |
| yDiff > NavigationBarCompat.getQuickScrubTouchSlopPx() && yDiff > xDiff; |
| exceededSwipeVerticalTouchSlop = |
| xDiff > NavigationBarCompat.getQuickStepTouchSlopPx() && xDiff > yDiff; |
| exceededSwipeVerticalDragSlop = |
| xDiff > NavigationBarCompat.getQuickStepDragSlopPx() && xDiff > yDiff; |
| posH = y; |
| touchDownH = mTouchDownY; |
| posV = x; |
| touchDownV = mTouchDownX; |
| } else { |
| exceededSwipeHorizontalTouchSlop = |
| xDiff > NavigationBarCompat.getQuickScrubTouchSlopPx() && xDiff > yDiff; |
| exceededSwipeVerticalTouchSlop = |
| yDiff > NavigationBarCompat.getQuickStepTouchSlopPx() && yDiff > xDiff; |
| exceededSwipeVerticalDragSlop = |
| yDiff > NavigationBarCompat.getQuickStepDragSlopPx() && yDiff > xDiff; |
| posH = x; |
| touchDownH = mTouchDownX; |
| posV = y; |
| touchDownV = mTouchDownY; |
| } |
| |
| if (mCurrentAction != null) { |
| // Gesture started, provide positions to the current action |
| mCurrentAction.onGestureMove(x, y); |
| } else { |
| // Detect gesture and try to execute an action, only one can run at a time |
| if (exceededSwipeVerticalTouchSlop || exceededSwipeVerticalDragSlop) { |
| if (mDragVPositive ? (posV < touchDownV) : (posV > touchDownV)) { |
| // Swipe up gesture must use the larger slop |
| if (exceededSwipeVerticalTouchSlop) { |
| // Swiping up gesture |
| tryToStartGesture(mGestureActions[ACTION_SWIPE_UP_INDEX], |
| false /* alignedWithNavBar */, event); |
| } |
| } else { |
| // Swiping down gesture |
| tryToStartGesture(mGestureActions[ACTION_SWIPE_DOWN_INDEX], |
| false /* alignedWithNavBar */, event); |
| } |
| } else if (exceededSwipeHorizontalTouchSlop) { |
| if (mDragHPositive ? (posH < touchDownH) : (posH > touchDownH)) { |
| // Swiping left (rtl) gesture |
| tryToStartGesture(mGestureActions[ACTION_SWIPE_LEFT_INDEX], |
| true /* alignedWithNavBar */, event); |
| } else { |
| // Swiping right (ltr) gesture |
| tryToStartGesture(mGestureActions[ACTION_SWIPE_RIGHT_INDEX], |
| true /* alignedWithNavBar */, event); |
| } |
| } |
| } |
| |
| handleDragHitTarget(mGestureHorizontalDragsButton ? posH : posV, |
| mGestureHorizontalDragsButton ? touchDownH : touchDownV); |
| break; |
| } |
| case MotionEvent.ACTION_CANCEL: |
| case MotionEvent.ACTION_UP: |
| if (mCurrentAction != null) { |
| mCurrentAction.endGesture(); |
| } else if (action == MotionEvent.ACTION_UP) { |
| if (canTriggerEdgeSwipe(event)) { |
| int index = mNavigationBarView.getWindowTarget() == NAV_BAR_LEFT |
| ? ACTION_SWIPE_RIGHT_FROM_EDGE_INDEX |
| : ACTION_SWIPE_LEFT_FROM_EDGE_INDEX; |
| tryToStartGesture(mGestureActions[index], false /* alignedWithNavBar */, |
| event); |
| if (mCurrentAction != null) { |
| mCurrentAction.endGesture(); |
| } |
| } else if (QuickStepContract.isNavBarClickThrough(mContext) |
| && !mClickThroughPressed) { |
| // Enable click through functionality where no gesture has been detected and |
| // not passed the drag slop so inject a touch event at the same location |
| // after making the navigation bar window untouchable. After a some time, |
| // the navigation bar will be able to take input events again |
| float diffX = Math.abs(event.getX() - mTouchDownX); |
| float diffY = Math.abs(event.getY() - mTouchDownY); |
| |
| if ((diffX <= NavigationBarCompat.getQuickStepDragSlopPx() |
| && diffY <= NavigationBarCompat.getQuickStepDragSlopPx())) { |
| mNavigationBarView.setWindowTouchable(false); |
| mClickThroughPressX = event.getRawX(); |
| mClickThroughPressY = event.getRawY(); |
| mClickThroughPressed = true; |
| mNavigationBarView.postDelayed(mClickThroughSendTap, |
| CLICK_THROUGH_TAP_DELAY); |
| } |
| } |
| } |
| |
| // Return the hit target back to its original position |
| if (mHitTarget != null) { |
| final View button = mHitTarget.getCurrentView(); |
| if (mGestureHorizontalDragsButton || mGestureVerticalDragsButton) { |
| mDragBtnAnimator = button.animate().setDuration(BACK_BUTTON_FADE_IN_ALPHA) |
| .setInterpolator(Interpolators.FAST_OUT_SLOW_IN); |
| if (mGestureVerticalDragsButton ^ isNavBarVertical()) { |
| mDragBtnAnimator.translationY(0); |
| } else { |
| mDragBtnAnimator.translationX(0); |
| } |
| mDragBtnAnimator.start(); |
| } |
| } |
| break; |
| } |
| |
| if (shouldProxyEvents(action)) { |
| proxyMotionEvents(event); |
| } |
| |
| // Clear action when gesture and event proxy finishes |
| if (action == MotionEvent.ACTION_CANCEL || action == MotionEvent.ACTION_UP) { |
| mCurrentAction = null; |
| } |
| return mCurrentAction != null || deadZoneConsumed; |
| } |
| |
| private void handleDragHitTarget(int position, int touchDown) { |
| // Drag the hit target if gesture action requires it |
| if (mHitTarget != null && (mGestureVerticalDragsButton || mGestureHorizontalDragsButton)) { |
| final View button = mHitTarget.getCurrentView(); |
| if (mDragBtnAnimator != null) { |
| mDragBtnAnimator.cancel(); |
| mDragBtnAnimator = null; |
| } |
| |
| // Clamp drag to the bounding box of the navigation bar |
| float diff = (position - touchDown) * mDragDampeningFactor; |
| diff = Utilities.clamp(diff, mMinDragLimit, mMaxDragLimit); |
| if (mGestureVerticalDragsButton ^ isNavBarVertical()) { |
| button.setTranslationY(diff); |
| } else { |
| button.setTranslationX(diff); |
| } |
| } |
| } |
| |
| private boolean shouldProxyEvents(int action) { |
| // Do not send events for side navigation bar panels |
| if (mNavigationBarView.getWindowTarget() != WINDOW_TARGET_BOTTOM) { |
| return false; |
| } |
| final boolean actionValid = (mCurrentAction == null |
| || !mCurrentAction.disableProxyEvents()); |
| if (actionValid && !mIsInScreenPinning) { |
| // Allow down, cancel and up events, move and other events are passed if notifications |
| // are not showing and disabled gestures (such as long press) are not executed |
| switch (action) { |
| case MotionEvent.ACTION_DOWN: |
| case MotionEvent.ACTION_CANCEL: |
| case MotionEvent.ACTION_UP: |
| return true; |
| default: |
| return !mNotificationsVisibleOnDown && mAllowGestureDetection; |
| } |
| } |
| return false; |
| } |
| |
| @Override |
| public void onDraw(Canvas canvas) { |
| if (mCurrentAction != null) { |
| mCurrentAction.onDraw(canvas); |
| } |
| } |
| |
| @Override |
| public void onLayout(boolean changed, int left, int top, int right, int bottom) { |
| for (NavigationGestureAction action: mGestureActions) { |
| if (action != null) { |
| action.onLayout(changed, left, top, right, bottom); |
| } |
| } |
| mLastLayoutRect.set(left, top, right, bottom); |
| } |
| |
| @Override |
| public void onDarkIntensityChange(float intensity) { |
| final float oldIntensity = mDarkIntensity; |
| mDarkIntensity = intensity; |
| |
| // When in quick scrub, invalidate gradient if changing intensity from black to white and |
| // vice-versa |
| if (mCurrentAction != null && mNavigationBarView.isQuickScrubEnabled() |
| && Math.round(intensity) != Math.round(oldIntensity)) { |
| mCurrentAction.onDarkIntensityChange(mDarkIntensity); |
| } |
| mNavigationBarView.invalidate(); |
| } |
| |
| @Override |
| public void setBarState(boolean isRTL, int navBarPosition) { |
| final boolean changed = (mIsRTL != isRTL) || (mNavBarPosition != navBarPosition); |
| mIsRTL = isRTL; |
| mNavBarPosition = navBarPosition; |
| |
| // Determine the drag directions depending on location of nav bar |
| switch (navBarPosition) { |
| case NAV_BAR_LEFT: |
| mDragHPositive = !isRTL; |
| mDragVPositive = false; |
| break; |
| case NAV_BAR_RIGHT: |
| mDragHPositive = isRTL; |
| mDragVPositive = true; |
| break; |
| case NAV_BAR_BOTTOM: |
| mDragHPositive = !isRTL; |
| mDragVPositive = true; |
| break; |
| case NAV_BAR_INVALID: |
| Log.e(TAG, "Invalid nav bar position"); |
| break; |
| } |
| |
| for (NavigationGestureAction action: mGestureActions) { |
| if (action != null) { |
| action.setBarState(changed, mNavBarPosition, mDragHPositive, mDragVPositive); |
| } |
| } |
| } |
| |
| @Override |
| public void onNavigationButtonLongPress(View v) { |
| mAllowGestureDetection = false; |
| } |
| |
| @Override |
| public void dump(PrintWriter pw) { |
| pw.println("QuickStepController {"); |
| pw.print(" "); pw.println("mAllowGestureDetection=" + mAllowGestureDetection); |
| pw.print(" "); pw.println("mNotificationsVisibleOnDown=" + mNotificationsVisibleOnDown); |
| pw.print(" "); pw.println("mNavBarPosition=" + mNavBarPosition); |
| pw.print(" "); pw.println("mIsRTL=" + mIsRTL); |
| pw.print(" "); pw.println("mIsInScreenPinning=" + mIsInScreenPinning); |
| pw.println("}"); |
| } |
| |
| public NavigationGestureAction getCurrentAction() { |
| return mCurrentAction; |
| } |
| |
| private void tryToStartGesture(NavigationGestureAction action, boolean alignedWithNavBar, |
| MotionEvent event) { |
| if (action == null) { |
| return; |
| } |
| if (mIsInScreenPinning) { |
| mNavigationBarView.showPinningEscapeToast(); |
| mAllowGestureDetection = false; |
| return; |
| } |
| |
| // Start new action from gesture if is able to start and depending on notifications |
| // visibility and starting touch down target. If the action is enabled, then also check if |
| // can perform the action so that if action requires the button to be dragged, then the |
| // gesture will have a large dampening factor and prevent action from running. |
| final boolean validHitTarget = action.requiresTouchDownHitTarget() == HIT_TARGET_NONE |
| || action.requiresTouchDownHitTarget() == mNavigationBarView.getDownHitTarget(); |
| if (mCurrentAction == null && validHitTarget && action.isEnabled() |
| && (!mNotificationsVisibleOnDown || action.canRunWhenNotificationsShowing())) { |
| if (action.canPerformAction()) { |
| mCurrentAction = action; |
| event.transform(mTransformGlobalMatrix); |
| action.startGesture(event); |
| event.transform(mTransformLocalMatrix); |
| |
| // Calculate the bounding limits of drag to avoid dragging off nav bar's window |
| if (action.allowHitTargetToMoveOverDrag() && mHitTarget != null) { |
| final int[] buttonCenter = new int[2]; |
| View button = mHitTarget.getCurrentView(); |
| button.getLocationInWindow(buttonCenter); |
| buttonCenter[0] += button.getWidth() / 2; |
| buttonCenter[1] += button.getHeight() / 2; |
| final int x = isNavBarVertical() ? buttonCenter[1] : buttonCenter[0]; |
| final int y = isNavBarVertical() ? buttonCenter[0] : buttonCenter[1]; |
| final int iconHalfSize = mContext.getResources() |
| .getDimensionPixelSize(R.dimen.navigation_icon_size) / 2; |
| |
| if (alignedWithNavBar) { |
| mMinDragLimit = iconHalfSize - x; |
| mMaxDragLimit = -x - iconHalfSize + (isNavBarVertical() |
| ? mNavigationBarView.getHeight() : mNavigationBarView.getWidth()); |
| } else { |
| mMinDragLimit = iconHalfSize - y; |
| mMaxDragLimit = -y - iconHalfSize + (isNavBarVertical() |
| ? mNavigationBarView.getWidth() : mNavigationBarView.getHeight()); |
| } |
| } |
| } |
| |
| // Handle direction of the hit target drag from the axis that started the gesture |
| // Also calculate the dampening factor, weaker dampening if there is an active action |
| if (action.allowHitTargetToMoveOverDrag()) { |
| if (alignedWithNavBar) { |
| mGestureHorizontalDragsButton = true; |
| mGestureVerticalDragsButton = false; |
| mDragDampeningFactor = action.isActive() |
| ? HORIZONTAL_GESTURE_DAMPING : HORIZONTAL_DISABLED_GESTURE_DAMPING; |
| } else { |
| mGestureVerticalDragsButton = true; |
| mGestureHorizontalDragsButton = false; |
| mDragDampeningFactor = action.isActive() |
| ? VERTICAL_GESTURE_DAMPING : VERTICAL_DISABLED_GESTURE_DAMPING; |
| } |
| } |
| |
| if (mHitTarget != null) { |
| mHitTarget.abortCurrentGesture(); |
| } |
| } |
| } |
| |
| /** |
| * To trigger an edge swipe, the user must start from the left or right edges of certain height |
| * from the bottom then past the drag slope towards the center of the screen, followed by either |
| * a timed trigger for fast swipes or distance if held on the screen longer. |
| * For time, user must swipe up quickly before the Tap Timeout (typically 100ms) and for |
| * distance, the user can drag back to cancel if the touch up has not past the threshold. |
| * @param event Touch up event |
| * @return whether or not edge swipe gesture occurs |
| */ |
| private boolean canTriggerEdgeSwipe(MotionEvent event) { |
| if (mNavigationBarView.getWindowTarget() == WINDOW_TARGET_BOTTOM) { |
| return false; |
| } |
| int x = (int) event.getX(); |
| int y = (int) event.getY(); |
| int xDiff = Math.abs(x - mTouchDownX); |
| int yDiff = Math.abs(y - mTouchDownY); |
| final boolean exceededSwipeTouchSlop = xDiff > NavigationBarCompat.getQuickStepDragSlopPx() |
| && xDiff > yDiff; |
| if (exceededSwipeTouchSlop) { |
| long timeDiff = event.getEventTime() - event.getDownTime(); |
| return xDiff > mGestureRegionThreshold || timeDiff < ViewConfiguration.getTapTimeout(); |
| } |
| return false; |
| } |
| |
| private boolean canPerformAnyAction() { |
| for (NavigationGestureAction action: mGestureActions) { |
| if (action != null && action.isEnabled()) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| private void sendTap(float x, float y) { |
| long now = SystemClock.uptimeMillis(); |
| injectMotionEvent(InputDevice.SOURCE_TOUCHSCREEN, MotionEvent.ACTION_DOWN, now, x, y, 1.0f); |
| injectMotionEvent(InputDevice.SOURCE_TOUCHSCREEN, MotionEvent.ACTION_UP, now, x, y, 0.0f); |
| } |
| |
| private int getInputDeviceId(int inputSource) { |
| int[] devIds = InputDevice.getDeviceIds(); |
| for (int devId : devIds) { |
| InputDevice inputDev = InputDevice.getDevice(devId); |
| if (inputDev.supportsSource(inputSource)) { |
| return devId; |
| } |
| } |
| return 0; |
| } |
| |
| private void injectMotionEvent(int inputSource, int action, long when, float x, float y, |
| float pressure) { |
| final float defaultSize = 1.0f; |
| final int defaultMetaState = 0; |
| final float defaultPrecisionX = 1.0f; |
| final float defaultPrecisionY = 1.0f; |
| final int defaultEdgeFlags = 0; |
| MotionEvent event = MotionEvent.obtain(when, when, action, x, y, pressure, defaultSize, |
| defaultMetaState, defaultPrecisionX, defaultPrecisionY, |
| getInputDeviceId(inputSource), defaultEdgeFlags); |
| event.setSource(inputSource); |
| InputManager.getInstance().injectInputEvent(event, |
| InputManager.INJECT_INPUT_EVENT_MODE_ASYNC); |
| } |
| |
| private boolean proxyMotionEvents(MotionEvent event) { |
| event.transform(mTransformGlobalMatrix); |
| InputEventDispatcher dispatcher = mOverviewEventSender.getInputEventDispatcher(); |
| if (dispatcher != null) { |
| dispatcher.dispatch(event); |
| } |
| |
| final IOverviewProxy overviewProxy = mOverviewEventSender.getProxy(); |
| try { |
| if (event.getActionMasked() == MotionEvent.ACTION_DOWN) { |
| overviewProxy.onPreMotionEvent(mNavigationBarView.getDownHitTarget()); |
| } |
| overviewProxy.onMotionEvent(event); |
| if (DEBUG_OVERVIEW_PROXY) { |
| Log.d(TAG_OPS, "Send MotionEvent: " + event.toString()); |
| } |
| return true; |
| } catch (RemoteException e) { |
| Log.e(TAG, "Callback failed", e); |
| } finally { |
| event.transform(mTransformLocalMatrix); |
| } |
| return false; |
| } |
| |
| protected boolean isNavBarVertical() { |
| return mNavBarPosition == NAV_BAR_LEFT || mNavBarPosition == NAV_BAR_RIGHT; |
| } |
| |
| private static int convertDpToPixel(float dp) { |
| return (int) (dp * Resources.getSystem().getDisplayMetrics().density); |
| } |
| |
| // TODO(112934365): Clean up following methods when cleaning up nav bar experiments |
| |
| static boolean getBoolGlobalSetting(Context context, String key) { |
| return Settings.Global.getInt(context.getContentResolver(), key, 0) != 0; |
| } |
| |
| static int getIntGlobalSetting(Context context, String key, int defaultValue) { |
| return Settings.Global.getInt(context.getContentResolver(), key, defaultValue); |
| } |
| |
| /** |
| * @return whether to hide the back button. |
| */ |
| public static boolean shouldHideBackButton(Context context) { |
| return QuickStepContract.isGesturalMode(context); |
| } |
| |
| /** |
| * @return whether to hide the home button. |
| */ |
| public static boolean shouldHideHomeButton(Context context) { |
| return QuickStepContract.isGesturalMode(context); |
| } |
| |
| /** |
| * @return whether to show the home handle. |
| */ |
| public static boolean showHomeHandle(Context context) { |
| return QuickStepContract.isGesturalMode(context); |
| } |
| } |