Merge "Add feature flag + leftward swipe for Compose access." into ub-launcher3-master

This commit is contained in:
James O'Leary 2019-10-08 19:03:04 +00:00 committed by Android (Google) Code Review
commit 8d46a3dc00
6 changed files with 242 additions and 1 deletions

View File

@ -53,6 +53,7 @@ import androidx.annotation.WorkerThread;
import com.android.launcher3.BaseDraggingActivity; import com.android.launcher3.BaseDraggingActivity;
import com.android.launcher3.Utilities; import com.android.launcher3.Utilities;
import com.android.launcher3.config.FeatureFlags;
import com.android.launcher3.logging.EventLogArray; import com.android.launcher3.logging.EventLogArray;
import com.android.launcher3.logging.UserEventDispatcher; import com.android.launcher3.logging.UserEventDispatcher;
import com.android.launcher3.model.AppLaunchTracker; import com.android.launcher3.model.AppLaunchTracker;
@ -69,6 +70,7 @@ import com.android.quickstep.inputconsumers.InputConsumer;
import com.android.quickstep.inputconsumers.OtherActivityInputConsumer; import com.android.quickstep.inputconsumers.OtherActivityInputConsumer;
import com.android.quickstep.inputconsumers.OverviewInputConsumer; import com.android.quickstep.inputconsumers.OverviewInputConsumer;
import com.android.quickstep.inputconsumers.OverviewWithoutFocusInputConsumer; import com.android.quickstep.inputconsumers.OverviewWithoutFocusInputConsumer;
import com.android.quickstep.inputconsumers.QuickCaptureTouchConsumer;
import com.android.quickstep.inputconsumers.ResetGestureInputConsumer; import com.android.quickstep.inputconsumers.ResetGestureInputConsumer;
import com.android.quickstep.inputconsumers.ScreenPinnedInputConsumer; import com.android.quickstep.inputconsumers.ScreenPinnedInputConsumer;
import com.android.systemui.shared.recents.IOverviewProxy; import com.android.systemui.shared.recents.IOverviewProxy;
@ -454,6 +456,13 @@ public class TouchInteractionService extends Service implements
mInputMonitorCompat); mInputMonitorCompat);
} }
if (FeatureFlags.ENABLE_QUICK_CAPTURE_GESTURE.get()) {
// Put the Compose gesture as higher priority than the Assistant or base gestures
base = new QuickCaptureTouchConsumer(this, base,
mInputMonitorCompat, mOverviewComponentObserver.getActivityControlHelper());
}
if (mDeviceState.isScreenPinningActive()) { if (mDeviceState.isScreenPinningActive()) {
// Note: we only allow accessibility to wrap this, and it replaces the previous // Note: we only allow accessibility to wrap this, and it replaces the previous
// base input consumer (which should be NO_OP anyway since topTaskLocked == true). // base input consumer (which should be NO_OP anyway since topTaskLocked == true).

View File

@ -131,8 +131,8 @@ public class AssistantInputConsumer extends DelegateInputConsumer {
case ACTION_POINTER_DOWN: { case ACTION_POINTER_DOWN: {
if (mState != STATE_ACTIVE) { if (mState != STATE_ACTIVE) {
mState = STATE_DELEGATE_ACTIVE; mState = STATE_DELEGATE_ACTIVE;
break;
} }
break;
} }
case ACTION_POINTER_UP: { case ACTION_POINTER_UP: {
int ptrIdx = ev.getActionIndex(); int ptrIdx = ev.getActionIndex();

View File

@ -33,6 +33,7 @@ public interface InputConsumer {
int TYPE_SCREEN_PINNED = 1 << 6; int TYPE_SCREEN_PINNED = 1 << 6;
int TYPE_OVERVIEW_WITHOUT_FOCUS = 1 << 7; int TYPE_OVERVIEW_WITHOUT_FOCUS = 1 << 7;
int TYPE_RESET_GESTURE = 1 << 8; int TYPE_RESET_GESTURE = 1 << 8;
int TYPE_QUICK_CAPTURE = 1 << 9;
String[] NAMES = new String[] { String[] NAMES = new String[] {
"TYPE_NO_OP", // 0 "TYPE_NO_OP", // 0
@ -44,6 +45,7 @@ public interface InputConsumer {
"TYPE_SCREEN_PINNED", // 6 "TYPE_SCREEN_PINNED", // 6
"TYPE_OVERVIEW_WITHOUT_FOCUS", // 7 "TYPE_OVERVIEW_WITHOUT_FOCUS", // 7
"TYPE_RESET_GESTURE", // 8 "TYPE_RESET_GESTURE", // 8
"TYPE_QUICK_CAPTURE", // 9
}; };
InputConsumer NO_OP = () -> TYPE_NO_OP; InputConsumer NO_OP = () -> TYPE_NO_OP;

View File

@ -0,0 +1,219 @@
/*
* Copyright (C) 2019 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.quickstep.inputconsumers;
import static android.view.MotionEvent.ACTION_CANCEL;
import static android.view.MotionEvent.ACTION_DOWN;
import static android.view.MotionEvent.ACTION_MOVE;
import static android.view.MotionEvent.ACTION_POINTER_DOWN;
import static android.view.MotionEvent.ACTION_POINTER_UP;
import static android.view.MotionEvent.ACTION_UP;
import static com.android.launcher3.Utilities.squaredHypot;
import android.app.ActivityOptions;
import android.content.Context;
import android.content.Intent;
import android.content.pm.PackageManager;
import android.graphics.PointF;
import android.os.Bundle;
import android.view.MotionEvent;
import android.view.ViewConfiguration;
import com.android.launcher3.BaseDraggingActivity;
import com.android.launcher3.R;
import com.android.quickstep.ActivityControlHelper;
import com.android.quickstep.views.RecentsView;
import com.android.systemui.shared.system.InputMonitorCompat;
/**
* Touch consumer for handling events to launch quick capture from launcher
* @param <T> Draggable activity subclass used by RecentsView
*/
public class QuickCaptureTouchConsumer<T extends BaseDraggingActivity>
extends DelegateInputConsumer {
private static final String TAG = "QuickCaptureTouchConsumer";
private static final String QUICK_CAPTURE_PACKAGE = "com.google.auxe.compose";
private static final String QUICK_CAPTURE_PACKAGE_DEV = "com.google.auxe.compose.debug";
private static final String EXTRA_DEVICE_STATE = "deviceState";
private static final String DEVICE_STATE_LOCKED = "Locked";
private static final String DEVICE_STATE_LAUNCHER = "Launcher";
private static final String DEVICE_STATE_APP = "App";
private static final String DEVICE_STATE_UNKNOWN = "Unknown";
private static final int ANGLE_THRESHOLD = 35; // Degrees
private final PointF mDownPos = new PointF();
private final PointF mLastPos = new PointF();
private final PointF mStartDragPos = new PointF();
private int mActivePointerId = -1;
private boolean mPassedSlop = false;
private final float mSquaredSlop;
private Context mContext;
private RecentsView mRecentsView;
public QuickCaptureTouchConsumer(Context context, InputConsumer delegate,
InputMonitorCompat inputMonitor, ActivityControlHelper<T> activityControlHelper) {
super(delegate, inputMonitor);
mContext = context;
float slop = ViewConfiguration.get(context).getScaledTouchSlop();
mSquaredSlop = slop * slop;
activityControlHelper.createActivityInitListener(this::onActivityInit).register();
}
@Override
public int getType() {
return TYPE_QUICK_CAPTURE | mDelegate.getType();
}
private boolean onActivityInit(final T activity, Boolean alreadyOnHome) {
mRecentsView = activity.getOverviewPanel();
return true;
}
@Override
public void onMotionEvent(MotionEvent ev) {
switch (ev.getActionMasked()) {
case ACTION_DOWN: {
mActivePointerId = ev.getPointerId(0);
mDownPos.set(ev.getX(), ev.getY());
mLastPos.set(mDownPos);
break;
}
case ACTION_POINTER_DOWN: {
if (mState != STATE_ACTIVE) {
mState = STATE_DELEGATE_ACTIVE;
}
break;
}
case ACTION_POINTER_UP: {
int ptrIdx = ev.getActionIndex();
int ptrId = ev.getPointerId(ptrIdx);
if (ptrId == mActivePointerId) {
final int newPointerIdx = ptrIdx == 0 ? 1 : 0;
mDownPos.set(
ev.getX(newPointerIdx) - (mLastPos.x - mDownPos.x),
ev.getY(newPointerIdx) - (mLastPos.y - mDownPos.y));
mLastPos.set(ev.getX(newPointerIdx), ev.getY(newPointerIdx));
mActivePointerId = ev.getPointerId(newPointerIdx);
}
break;
}
case ACTION_MOVE: {
if (mState == STATE_DELEGATE_ACTIVE) {
break;
}
if (!mDelegate.allowInterceptByParent()) {
mState = STATE_DELEGATE_ACTIVE;
break;
}
int pointerIndex = ev.findPointerIndex(mActivePointerId);
if (pointerIndex == -1) {
break;
}
mLastPos.set(ev.getX(pointerIndex), ev.getY(pointerIndex));
if (!mPassedSlop) {
// Normal gesture, ensure we pass the slop before we start tracking the gesture
if (squaredHypot(mLastPos.x - mDownPos.x, mLastPos.y - mDownPos.y)
> mSquaredSlop) {
mPassedSlop = true;
mStartDragPos.set(mLastPos.x, mLastPos.y);
if (isValidQuickCaptureGesture()) {
setActive(ev);
} else {
mState = STATE_DELEGATE_ACTIVE;
}
}
}
break;
}
case ACTION_CANCEL:
case ACTION_UP:
if (mState != STATE_DELEGATE_ACTIVE && mPassedSlop) {
startQuickCapture();
}
mPassedSlop = false;
mState = STATE_INACTIVE;
break;
}
if (mState != STATE_ACTIVE) {
mDelegate.onMotionEvent(ev);
}
}
private boolean isValidQuickCaptureGesture() {
// Make sure there isn't an app to quick switch to on our right
boolean atRightMostApp = (mRecentsView == null || mRecentsView.getRunningTaskIndex() <= 0);
// Check if the gesture is within our angle threshold of horizontal
float deltaY = Math.abs(mLastPos.y - mDownPos.y);
float deltaX = mDownPos.x - mLastPos.x; // Positive if this is a gesture to the left
boolean angleInBounds = Math.toDegrees(Math.atan2(deltaY, deltaX)) < ANGLE_THRESHOLD;
return atRightMostApp && angleInBounds;
}
private void startQuickCapture() {
// Inspect our delegate's type to figure out where the user invoked Compose
String deviceState = DEVICE_STATE_UNKNOWN;
int consumerType = mDelegate.getType();
if (((consumerType & InputConsumer.TYPE_OVERVIEW) > 0)
|| ((consumerType & InputConsumer.TYPE_OVERVIEW_WITHOUT_FOCUS)) > 0) {
deviceState = DEVICE_STATE_LAUNCHER;
} else if ((consumerType & InputConsumer.TYPE_OTHER_ACTIVITY) > 0) {
deviceState = DEVICE_STATE_APP;
} else if (((consumerType & InputConsumer.TYPE_RESET_GESTURE) > 0)
|| ((consumerType & InputConsumer.TYPE_DEVICE_LOCKED) > 0)) {
deviceState = DEVICE_STATE_LOCKED;
}
// Then launch the app
PackageManager pm = mContext.getPackageManager();
Intent qcIntent = pm.getLaunchIntentForPackage(QUICK_CAPTURE_PACKAGE);
if (qcIntent == null) {
// If we couldn't find the regular app, try the dev version
qcIntent = pm.getLaunchIntentForPackage(QUICK_CAPTURE_PACKAGE_DEV);
}
if (qcIntent != null) {
qcIntent.putExtra(EXTRA_DEVICE_STATE, deviceState);
Bundle options = ActivityOptions.makeCustomAnimation(mContext, R.anim.slide_in_right,
0).toBundle();
mContext.startActivity(qcIntent, options);
}
}
}

View File

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
android:shareInterpolator="false" >
<translate
android:duration="@android:integer/config_shortAnimTime"
android:fromXDelta="100%"
android:toXDelta="0%"
/>
</set>

View File

@ -117,6 +117,8 @@ public final class FeatureFlags {
public static final TogglableFlag ENABLE_PREDICTION_DISMISS = new TogglableFlag( public static final TogglableFlag ENABLE_PREDICTION_DISMISS = new TogglableFlag(
"ENABLE_PREDICTION_DISMISS", false, "Allow option to dimiss apps from predicted list"); "ENABLE_PREDICTION_DISMISS", false, "Allow option to dimiss apps from predicted list");
public static final TogglableFlag ENABLE_QUICK_CAPTURE_GESTURE = new TogglableFlag(
"ENABLE_QUICK_CAPTURE_GESTURE", false, "Swipe from right to left to quick capture");
public static void initialize(Context context) { public static void initialize(Context context) {
// Avoid the disk read for user builds // Avoid the disk read for user builds