/* * Copyright (C) 2012 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; import static com.android.internal.jank.InteractionJankMonitor.CUJ_NOTIFICATION_SHADE_ROW_EXPAND; import android.animation.Animator; import android.animation.AnimatorListenerAdapter; import android.animation.ObjectAnimator; import android.content.Context; import android.util.FloatProperty; import android.util.Log; import android.view.Gravity; import android.view.HapticFeedbackConstants; import android.view.MotionEvent; import android.view.ScaleGestureDetector; import android.view.ScaleGestureDetector.OnScaleGestureListener; import android.view.VelocityTracker; import android.view.View; import android.view.ViewConfiguration; import com.android.internal.annotations.VisibleForTesting; import com.android.internal.jank.InteractionJankMonitor; import com.android.systemui.statusbar.notification.row.ExpandableNotificationRow; import com.android.systemui.statusbar.notification.row.ExpandableView; import com.android.systemui.statusbar.policy.ScrollAdapter; import com.android.wm.shell.animation.FlingAnimationUtils; public class ExpandHelper implements Gefingerpoken { public interface Callback { ExpandableView getChildAtRawPosition(float x, float y); ExpandableView getChildAtPosition(float x, float y); boolean canChildBeExpanded(View v); void setUserExpandedChild(View v, boolean userExpanded); void setUserLockedChild(View v, boolean userLocked); void expansionStateChanged(boolean isExpanding); int getMaxExpandHeight(ExpandableView view); void setExpansionCancelled(View view); } private static final String TAG = "ExpandHelper"; protected static final boolean DEBUG = false; protected static final boolean DEBUG_SCALE = false; private static final float EXPAND_DURATION = 0.3f; // Set to false to disable focus-based gestures (spread-finger vertical pull). private static final boolean USE_DRAG = true; // Set to false to disable scale-based gestures (both horizontal and vertical). private static final boolean USE_SPAN = true; // Both gestures types may be active at the same time. // At least one gesture type should be active. // A variant of the screwdriver gesture will emerge from either gesture type. // amount of overstretch for maximum brightness expressed in U // 2f: maximum brightness is stretching a 1U to 3U, or a 4U to 6U private static final float STRETCH_INTERVAL = 2f; private static final FloatProperty VIEW_SCALER_HEIGHT_PROPERTY = new FloatProperty("ViewScalerHeight") { @Override public void setValue(ViewScaler object, float value) { object.setHeight(value); } @Override public Float get(ViewScaler object) { return object.getHeight(); } }; @SuppressWarnings("unused") private Context mContext; private boolean mExpanding; private static final int NONE = 0; private static final int BLINDS = 1<<0; private static final int PULL = 1<<1; private static final int STRETCH = 1<<2; private int mExpansionStyle = NONE; private boolean mWatchingForPull; private boolean mHasPopped; private View mEventSource; private float mOldHeight; private float mNaturalHeight; private float mInitialTouchFocusY; private float mInitialTouchX; private float mInitialTouchY; private float mInitialTouchSpan; private float mLastFocusY; private float mLastSpanY; private final int mTouchSlop; private final float mSlopMultiplier; private float mLastMotionY; private float mPullGestureMinXSpan; private Callback mCallback; private ScaleGestureDetector mSGD; private ViewScaler mScaler; private ObjectAnimator mScaleAnimation; private boolean mEnabled = true; private ExpandableView mResizedView; private float mCurrentHeight; private int mSmallSize; private int mLargeSize; private float mMaximumStretch; private boolean mOnlyMovements; private int mGravity; private ScrollAdapter mScrollAdapter; private FlingAnimationUtils mFlingAnimationUtils; private VelocityTracker mVelocityTracker; private OnScaleGestureListener mScaleGestureListener = new ScaleGestureDetector.SimpleOnScaleGestureListener() { @Override public boolean onScaleBegin(ScaleGestureDetector detector) { if (DEBUG_SCALE) Log.v(TAG, "onscalebegin()"); if (!mOnlyMovements) { startExpanding(mResizedView, STRETCH); } return mExpanding; } @Override public boolean onScale(ScaleGestureDetector detector) { if (DEBUG_SCALE) Log.v(TAG, "onscale() on " + mResizedView); return true; } @Override public void onScaleEnd(ScaleGestureDetector detector) { } }; @VisibleForTesting ObjectAnimator getScaleAnimation() { return mScaleAnimation; } private class ViewScaler { ExpandableView mView; public ViewScaler() {} public void setView(ExpandableView v) { mView = v; } public void setHeight(float h) { if (DEBUG_SCALE) Log.v(TAG, "SetHeight: setting to " + h); mView.setActualHeight((int) h); mCurrentHeight = h; } public float getHeight() { return mView.getActualHeight(); } public int getNaturalHeight() { return mCallback.getMaxExpandHeight(mView); } } /** * Handle expansion gestures to expand and contract children of the callback. * * @param context application context * @param callback the container that holds the items to be manipulated * @param small the smallest allowable size for the manipulated items. * @param large the largest allowable size for the manipulated items. */ public ExpandHelper(Context context, Callback callback, int small, int large) { mSmallSize = small; mMaximumStretch = mSmallSize * STRETCH_INTERVAL; mLargeSize = large; mContext = context; mCallback = callback; mScaler = new ViewScaler(); mGravity = Gravity.TOP; mScaleAnimation = ObjectAnimator.ofFloat(mScaler, VIEW_SCALER_HEIGHT_PROPERTY, 0f); mPullGestureMinXSpan = mContext.getResources().getDimension(R.dimen.pull_span_min); final ViewConfiguration configuration = ViewConfiguration.get(mContext); mTouchSlop = configuration.getScaledTouchSlop(); mSlopMultiplier = configuration.getAmbiguousGestureMultiplier(); mSGD = new ScaleGestureDetector(context, mScaleGestureListener); mFlingAnimationUtils = new FlingAnimationUtils(mContext.getResources().getDisplayMetrics(), EXPAND_DURATION); } @VisibleForTesting void updateExpansion() { if (DEBUG_SCALE) Log.v(TAG, "updateExpansion()"); // are we scaling or dragging? float span = mSGD.getCurrentSpan() - mInitialTouchSpan; span *= USE_SPAN ? 1f : 0f; float drag = mSGD.getFocusY() - mInitialTouchFocusY; drag *= USE_DRAG ? 1f : 0f; drag *= mGravity == Gravity.BOTTOM ? -1f : 1f; float pull = Math.abs(drag) + Math.abs(span) + 1f; float hand = drag * Math.abs(drag) / pull + span * Math.abs(span) / pull; float target = hand + mOldHeight; float newHeight = clamp(target); mScaler.setHeight(newHeight); mLastFocusY = mSGD.getFocusY(); mLastSpanY = mSGD.getCurrentSpan(); } private float clamp(float target) { float out = target; out = out < mSmallSize ? mSmallSize : out; out = out > mNaturalHeight ? mNaturalHeight : out; return out; } private ExpandableView findView(float x, float y) { ExpandableView v; if (mEventSource != null) { int[] location = new int[2]; mEventSource.getLocationOnScreen(location); x += location[0]; y += location[1]; v = mCallback.getChildAtRawPosition(x, y); } else { v = mCallback.getChildAtPosition(x, y); } return v; } private boolean isInside(View v, float x, float y) { if (DEBUG) Log.d(TAG, "isinside (" + x + ", " + y + ")"); if (v == null) { if (DEBUG) Log.d(TAG, "isinside null subject"); return false; } if (mEventSource != null) { int[] location = new int[2]; mEventSource.getLocationOnScreen(location); x += location[0]; y += location[1]; if (DEBUG) Log.d(TAG, " to global (" + x + ", " + y + ")"); } int[] location = new int[2]; v.getLocationOnScreen(location); x -= location[0]; y -= location[1]; if (DEBUG) Log.d(TAG, " to local (" + x + ", " + y + ")"); if (DEBUG) Log.d(TAG, " inside (" + v.getWidth() + ", " + v.getHeight() + ")"); boolean inside = (x > 0f && y > 0f && x < v.getWidth() & y < v.getHeight()); return inside; } public void setEventSource(View eventSource) { mEventSource = eventSource; } public void setGravity(int gravity) { mGravity = gravity; } public void setScrollAdapter(ScrollAdapter adapter) { mScrollAdapter = adapter; } private float getTouchSlop(MotionEvent event) { // Adjust the touch slop if another gesture may be being performed. return event.getClassification() == MotionEvent.CLASSIFICATION_AMBIGUOUS_GESTURE ? mTouchSlop * mSlopMultiplier : mTouchSlop; } @Override public boolean onInterceptTouchEvent(MotionEvent ev) { if (!isEnabled()) { return false; } trackVelocity(ev); final int action = ev.getAction(); if (DEBUG_SCALE) Log.d(TAG, "intercept: act=" + MotionEvent.actionToString(action) + " expanding=" + mExpanding + (0 != (mExpansionStyle & BLINDS) ? " (blinds)" : "") + (0 != (mExpansionStyle & PULL) ? " (pull)" : "") + (0 != (mExpansionStyle & STRETCH) ? " (stretch)" : "")); // check for a spread-finger vertical pull gesture mSGD.onTouchEvent(ev); final int x = (int) mSGD.getFocusX(); final int y = (int) mSGD.getFocusY(); mInitialTouchFocusY = y; mInitialTouchSpan = mSGD.getCurrentSpan(); mLastFocusY = mInitialTouchFocusY; mLastSpanY = mInitialTouchSpan; if (DEBUG_SCALE) Log.d(TAG, "set initial span: " + mInitialTouchSpan); if (mExpanding) { mLastMotionY = ev.getRawY(); maybeRecycleVelocityTracker(ev); return true; } else { if ((action == MotionEvent.ACTION_MOVE) && 0 != (mExpansionStyle & BLINDS)) { // we've begun Venetian blinds style expansion return true; } switch (action & MotionEvent.ACTION_MASK) { case MotionEvent.ACTION_MOVE: { final float xspan = mSGD.getCurrentSpanX(); if (xspan > mPullGestureMinXSpan && xspan > mSGD.getCurrentSpanY() && !mExpanding) { // detect a vertical pulling gesture with fingers somewhat separated if (DEBUG_SCALE) Log.v(TAG, "got pull gesture (xspan=" + xspan + "px)"); startExpanding(mResizedView, PULL); mWatchingForPull = false; } if (mWatchingForPull) { final float yDiff = ev.getRawY() - mInitialTouchY; final float xDiff = ev.getRawX() - mInitialTouchX; if (yDiff > getTouchSlop(ev) && yDiff > Math.abs(xDiff)) { if (DEBUG) Log.v(TAG, "got venetian gesture (dy=" + yDiff + "px)"); mWatchingForPull = false; if (mResizedView != null && !isFullyExpanded(mResizedView)) { if (startExpanding(mResizedView, BLINDS)) { mLastMotionY = ev.getRawY(); mInitialTouchY = ev.getRawY(); mHasPopped = false; } } } } break; } case MotionEvent.ACTION_DOWN: mWatchingForPull = mScrollAdapter != null && isInside(mScrollAdapter.getHostView(), x, y) && mScrollAdapter.isScrolledToTop(); mResizedView = findView(x, y); if (mResizedView != null && !mCallback.canChildBeExpanded(mResizedView)) { mResizedView = null; mWatchingForPull = false; } mInitialTouchY = ev.getRawY(); mInitialTouchX = ev.getRawX(); break; case MotionEvent.ACTION_CANCEL: case MotionEvent.ACTION_UP: if (DEBUG) Log.d(TAG, "up/cancel"); finishExpanding(ev.getActionMasked() == MotionEvent.ACTION_CANCEL /* forceAbort */, getCurrentVelocity()); clearView(); break; } mLastMotionY = ev.getRawY(); maybeRecycleVelocityTracker(ev); return mExpanding; } } private void trackVelocity(MotionEvent event) { int action = event.getActionMasked(); switch(action) { case MotionEvent.ACTION_DOWN: if (mVelocityTracker == null) { mVelocityTracker = VelocityTracker.obtain(); } else { mVelocityTracker.clear(); } mVelocityTracker.addMovement(event); break; case MotionEvent.ACTION_MOVE: if (mVelocityTracker == null) { mVelocityTracker = VelocityTracker.obtain(); } mVelocityTracker.addMovement(event); break; default: break; } } private void maybeRecycleVelocityTracker(MotionEvent event) { if (mVelocityTracker != null && (event.getActionMasked() == MotionEvent.ACTION_CANCEL || event.getActionMasked() == MotionEvent.ACTION_UP)) { mVelocityTracker.recycle(); mVelocityTracker = null; } } private float getCurrentVelocity() { if (mVelocityTracker != null) { mVelocityTracker.computeCurrentVelocity(1000); return mVelocityTracker.getYVelocity(); } else { return 0f; } } public void setEnabled(boolean enable) { mEnabled = enable; } private boolean isEnabled() { return mEnabled; } private boolean isFullyExpanded(ExpandableView underFocus) { return underFocus.getIntrinsicHeight() == underFocus.getMaxContentHeight() && (!underFocus.isSummaryWithChildren() || underFocus.areChildrenExpanded()); } @Override public boolean onTouchEvent(MotionEvent ev) { if (!isEnabled() && !mExpanding) { // In case we're expanding we still want to finish the current motion. return false; } trackVelocity(ev); final int action = ev.getActionMasked(); if (DEBUG_SCALE) Log.d(TAG, "touch: act=" + MotionEvent.actionToString(action) + " expanding=" + mExpanding + (0 != (mExpansionStyle & BLINDS) ? " (blinds)" : "") + (0 != (mExpansionStyle & PULL) ? " (pull)" : "") + (0 != (mExpansionStyle & STRETCH) ? " (stretch)" : "")); mSGD.onTouchEvent(ev); final int x = (int) mSGD.getFocusX(); final int y = (int) mSGD.getFocusY(); if (mOnlyMovements) { mLastMotionY = ev.getRawY(); return false; } switch (action) { case MotionEvent.ACTION_DOWN: mWatchingForPull = mScrollAdapter != null && isInside(mScrollAdapter.getHostView(), x, y); mResizedView = findView(x, y); mInitialTouchX = ev.getRawX(); mInitialTouchY = ev.getRawY(); break; case MotionEvent.ACTION_MOVE: { if (mWatchingForPull) { final float yDiff = ev.getRawY() - mInitialTouchY; final float xDiff = ev.getRawX() - mInitialTouchX; if (yDiff > getTouchSlop(ev) && yDiff > Math.abs(xDiff)) { if (DEBUG) Log.v(TAG, "got venetian gesture (dy=" + yDiff + "px)"); mWatchingForPull = false; if (mResizedView != null && !isFullyExpanded(mResizedView)) { if (startExpanding(mResizedView, BLINDS)) { mInitialTouchY = ev.getRawY(); mLastMotionY = ev.getRawY(); mHasPopped = false; } } } } if (mExpanding && 0 != (mExpansionStyle & BLINDS)) { final float rawHeight = ev.getRawY() - mLastMotionY + mCurrentHeight; final float newHeight = clamp(rawHeight); boolean isFinished = false; boolean expanded = false; if (rawHeight > mNaturalHeight) { isFinished = true; expanded = true; } if (rawHeight < mSmallSize) { isFinished = true; expanded = false; } if (!mHasPopped) { if (mEventSource != null) { mEventSource.performHapticFeedback(HapticFeedbackConstants.VIRTUAL_KEY); } mHasPopped = true; } mScaler.setHeight(newHeight); mLastMotionY = ev.getRawY(); if (isFinished) { mCallback.expansionStateChanged(false); } else { mCallback.expansionStateChanged(true); } return true; } if (mExpanding) { // Gestural expansion is running updateExpansion(); mLastMotionY = ev.getRawY(); return true; } break; } case MotionEvent.ACTION_POINTER_UP: case MotionEvent.ACTION_POINTER_DOWN: if (DEBUG) Log.d(TAG, "pointer change"); mInitialTouchY += mSGD.getFocusY() - mLastFocusY; mInitialTouchSpan += mSGD.getCurrentSpan() - mLastSpanY; break; case MotionEvent.ACTION_UP: case MotionEvent.ACTION_CANCEL: if (DEBUG) Log.d(TAG, "up/cancel"); finishExpanding(!isEnabled() || ev.getActionMasked() == MotionEvent.ACTION_CANCEL, getCurrentVelocity()); clearView(); break; } mLastMotionY = ev.getRawY(); maybeRecycleVelocityTracker(ev); return mResizedView != null; } /** * @return True if the view is expandable, false otherwise. */ @VisibleForTesting boolean startExpanding(ExpandableView v, int expandType) { if (!(v instanceof ExpandableNotificationRow)) { return false; } mExpansionStyle = expandType; if (mExpanding && v == mResizedView) { return true; } mExpanding = true; mCallback.expansionStateChanged(true); if (DEBUG) Log.d(TAG, "scale type " + expandType + " beginning on view: " + v); mCallback.setUserLockedChild(v, true); mScaler.setView(v); mOldHeight = mScaler.getHeight(); mCurrentHeight = mOldHeight; boolean canBeExpanded = mCallback.canChildBeExpanded(v); if (canBeExpanded) { if (DEBUG) Log.d(TAG, "working on an expandable child"); mNaturalHeight = mScaler.getNaturalHeight(); mSmallSize = v.getCollapsedHeight(); } else { if (DEBUG) Log.d(TAG, "working on a non-expandable child"); mNaturalHeight = mOldHeight; } if (DEBUG) Log.d(TAG, "got mOldHeight: " + mOldHeight + " mNaturalHeight: " + mNaturalHeight); InteractionJankMonitor.getInstance().begin(v, CUJ_NOTIFICATION_SHADE_ROW_EXPAND); return true; } /** * Finish the current expand motion * @param forceAbort whether the expansion should be forcefully aborted and returned to the old * state * @param velocity the velocity this was expanded/ collapsed with */ @VisibleForTesting void finishExpanding(boolean forceAbort, float velocity) { finishExpanding(forceAbort, velocity, true /* allowAnimation */); } /** * Finish the current expand motion * @param forceAbort whether the expansion should be forcefully aborted and returned to the old * state * @param velocity the velocity this was expanded/ collapsed with */ private void finishExpanding(boolean forceAbort, float velocity, boolean allowAnimation) { if (!mExpanding) return; if (DEBUG) Log.d(TAG, "scale in finishing on view: " + mResizedView); float currentHeight = mScaler.getHeight(); final boolean wasClosed = (mOldHeight == mSmallSize); boolean nowExpanded; if (!forceAbort) { if (wasClosed) { nowExpanded = currentHeight > mOldHeight && velocity >= 0; } else { nowExpanded = currentHeight >= mOldHeight || velocity > 0; } nowExpanded |= mNaturalHeight == mSmallSize; } else { nowExpanded = !wasClosed; } if (mScaleAnimation.isRunning()) { mScaleAnimation.cancel(); } mCallback.expansionStateChanged(false); int naturalHeight = mScaler.getNaturalHeight(); float targetHeight = nowExpanded ? naturalHeight : mSmallSize; if (targetHeight != currentHeight && mEnabled && allowAnimation) { mScaleAnimation.setFloatValues(targetHeight); mScaleAnimation.setupStartValues(); final View scaledView = mResizedView; final boolean expand = nowExpanded; mScaleAnimation.addListener(new AnimatorListenerAdapter() { public boolean mCancelled; @Override public void onAnimationEnd(Animator animation) { if (!mCancelled) { mCallback.setUserExpandedChild(scaledView, expand); if (!mExpanding) { mScaler.setView(null); } } else { mCallback.setExpansionCancelled(scaledView); } mCallback.setUserLockedChild(scaledView, false); mScaleAnimation.removeListener(this); if (wasClosed) { InteractionJankMonitor.getInstance().end(CUJ_NOTIFICATION_SHADE_ROW_EXPAND); } } @Override public void onAnimationCancel(Animator animation) { mCancelled = true; } }); velocity = nowExpanded == velocity >= 0 ? velocity : 0; mFlingAnimationUtils.apply(mScaleAnimation, currentHeight, targetHeight, velocity); mScaleAnimation.start(); } else { if (targetHeight != currentHeight) { mScaler.setHeight(targetHeight); } mCallback.setUserExpandedChild(mResizedView, nowExpanded); mCallback.setUserLockedChild(mResizedView, false); mScaler.setView(null); if (wasClosed) { InteractionJankMonitor.getInstance().end(CUJ_NOTIFICATION_SHADE_ROW_EXPAND); } } mExpanding = false; mExpansionStyle = NONE; if (DEBUG) Log.d(TAG, "wasClosed is: " + wasClosed); if (DEBUG) Log.d(TAG, "currentHeight is: " + currentHeight); if (DEBUG) Log.d(TAG, "mSmallSize is: " + mSmallSize); if (DEBUG) Log.d(TAG, "targetHeight is: " + targetHeight); if (DEBUG) Log.d(TAG, "scale was finished on view: " + mResizedView); } private void clearView() { mResizedView = null; } /** * Use this to abort any pending expansions in progress and force that there will be no * animations. */ public void cancelImmediately() { cancel(false /* allowAnimation */); } /** * Use this to abort any pending expansions in progress. */ public void cancel() { cancel(true /* allowAnimation */); } private void cancel(boolean allowAnimation) { finishExpanding(true /* forceAbort */, 0f /* velocity */, allowAnimation); clearView(); // reset the gesture detector mSGD = new ScaleGestureDetector(mContext, mScaleGestureListener); } /** * Change the expansion mode to only observe movements and don't perform any resizing. * This is needed when the expanding is finished and the scroller kicks in, * performing an overscroll motion. We only want to shrink it again when we are not * overscrolled. * * @param onlyMovements Should only movements be observed? */ public void onlyObserveMovements(boolean onlyMovements) { mOnlyMovements = onlyMovements; } }