1 /* 2 * Copyright (C) 2012 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 18 package com.android.systemui; 19 20 import android.animation.Animator; 21 import android.animation.AnimatorListenerAdapter; 22 import android.animation.ObjectAnimator; 23 import android.content.Context; 24 import android.util.Log; 25 import android.view.Gravity; 26 import android.view.HapticFeedbackConstants; 27 import android.view.MotionEvent; 28 import android.view.ScaleGestureDetector; 29 import android.view.ScaleGestureDetector.OnScaleGestureListener; 30 import android.view.VelocityTracker; 31 import android.view.View; 32 import android.view.ViewConfiguration; 33 34 import com.android.internal.annotations.VisibleForTesting; 35 import com.android.systemui.statusbar.FlingAnimationUtils; 36 import com.android.systemui.statusbar.notification.row.ExpandableNotificationRow; 37 import com.android.systemui.statusbar.notification.row.ExpandableView; 38 import com.android.systemui.statusbar.policy.ScrollAdapter; 39 40 public class ExpandHelper implements Gefingerpoken { 41 public interface Callback { getChildAtRawPosition(float x, float y)42 ExpandableView getChildAtRawPosition(float x, float y); getChildAtPosition(float x, float y)43 ExpandableView getChildAtPosition(float x, float y); canChildBeExpanded(View v)44 boolean canChildBeExpanded(View v); setUserExpandedChild(View v, boolean userExpanded)45 void setUserExpandedChild(View v, boolean userExpanded); setUserLockedChild(View v, boolean userLocked)46 void setUserLockedChild(View v, boolean userLocked); expansionStateChanged(boolean isExpanding)47 void expansionStateChanged(boolean isExpanding); getMaxExpandHeight(ExpandableView view)48 int getMaxExpandHeight(ExpandableView view); setExpansionCancelled(View view)49 void setExpansionCancelled(View view); 50 } 51 52 private static final String TAG = "ExpandHelper"; 53 protected static final boolean DEBUG = false; 54 protected static final boolean DEBUG_SCALE = false; 55 private static final float EXPAND_DURATION = 0.3f; 56 57 // Set to false to disable focus-based gestures (spread-finger vertical pull). 58 private static final boolean USE_DRAG = true; 59 // Set to false to disable scale-based gestures (both horizontal and vertical). 60 private static final boolean USE_SPAN = true; 61 // Both gestures types may be active at the same time. 62 // At least one gesture type should be active. 63 // A variant of the screwdriver gesture will emerge from either gesture type. 64 65 // amount of overstretch for maximum brightness expressed in U 66 // 2f: maximum brightness is stretching a 1U to 3U, or a 4U to 6U 67 private static final float STRETCH_INTERVAL = 2f; 68 69 @SuppressWarnings("unused") 70 private Context mContext; 71 72 private boolean mExpanding; 73 private static final int NONE = 0; 74 private static final int BLINDS = 1<<0; 75 private static final int PULL = 1<<1; 76 private static final int STRETCH = 1<<2; 77 private int mExpansionStyle = NONE; 78 private boolean mWatchingForPull; 79 private boolean mHasPopped; 80 private View mEventSource; 81 private float mOldHeight; 82 private float mNaturalHeight; 83 private float mInitialTouchFocusY; 84 private float mInitialTouchX; 85 private float mInitialTouchY; 86 private float mInitialTouchSpan; 87 private float mLastFocusY; 88 private float mLastSpanY; 89 private final int mTouchSlop; 90 private final float mSlopMultiplier; 91 private float mLastMotionY; 92 private float mPullGestureMinXSpan; 93 private Callback mCallback; 94 private ScaleGestureDetector mSGD; 95 private ViewScaler mScaler; 96 private ObjectAnimator mScaleAnimation; 97 private boolean mEnabled = true; 98 private ExpandableView mResizedView; 99 private float mCurrentHeight; 100 101 private int mSmallSize; 102 private int mLargeSize; 103 private float mMaximumStretch; 104 private boolean mOnlyMovements; 105 106 private int mGravity; 107 108 private ScrollAdapter mScrollAdapter; 109 private FlingAnimationUtils mFlingAnimationUtils; 110 private VelocityTracker mVelocityTracker; 111 112 private OnScaleGestureListener mScaleGestureListener 113 = new ScaleGestureDetector.SimpleOnScaleGestureListener() { 114 @Override 115 public boolean onScaleBegin(ScaleGestureDetector detector) { 116 if (DEBUG_SCALE) Log.v(TAG, "onscalebegin()"); 117 118 if (!mOnlyMovements) { 119 startExpanding(mResizedView, STRETCH); 120 } 121 return mExpanding; 122 } 123 124 @Override 125 public boolean onScale(ScaleGestureDetector detector) { 126 if (DEBUG_SCALE) Log.v(TAG, "onscale() on " + mResizedView); 127 return true; 128 } 129 130 @Override 131 public void onScaleEnd(ScaleGestureDetector detector) { 132 } 133 }; 134 135 @VisibleForTesting getScaleAnimation()136 ObjectAnimator getScaleAnimation() { 137 return mScaleAnimation; 138 } 139 140 private class ViewScaler { 141 ExpandableView mView; 142 ViewScaler()143 public ViewScaler() {} setView(ExpandableView v)144 public void setView(ExpandableView v) { 145 mView = v; 146 } setHeight(float h)147 public void setHeight(float h) { 148 if (DEBUG_SCALE) Log.v(TAG, "SetHeight: setting to " + h); 149 mView.setActualHeight((int) h); 150 mCurrentHeight = h; 151 } getHeight()152 public float getHeight() { 153 return mView.getActualHeight(); 154 } getNaturalHeight()155 public int getNaturalHeight() { 156 return mCallback.getMaxExpandHeight(mView); 157 } 158 } 159 160 /** 161 * Handle expansion gestures to expand and contract children of the callback. 162 * 163 * @param context application context 164 * @param callback the container that holds the items to be manipulated 165 * @param small the smallest allowable size for the manuipulated items. 166 * @param large the largest allowable size for the manuipulated items. 167 */ ExpandHelper(Context context, Callback callback, int small, int large)168 public ExpandHelper(Context context, Callback callback, int small, int large) { 169 mSmallSize = small; 170 mMaximumStretch = mSmallSize * STRETCH_INTERVAL; 171 mLargeSize = large; 172 mContext = context; 173 mCallback = callback; 174 mScaler = new ViewScaler(); 175 mGravity = Gravity.TOP; 176 mScaleAnimation = ObjectAnimator.ofFloat(mScaler, "height", 0f); 177 mPullGestureMinXSpan = mContext.getResources().getDimension(R.dimen.pull_span_min); 178 179 final ViewConfiguration configuration = ViewConfiguration.get(mContext); 180 mTouchSlop = configuration.getScaledTouchSlop(); 181 mSlopMultiplier = configuration.getAmbiguousGestureMultiplier(); 182 183 mSGD = new ScaleGestureDetector(context, mScaleGestureListener); 184 mFlingAnimationUtils = new FlingAnimationUtils(mContext.getResources().getDisplayMetrics(), 185 EXPAND_DURATION); 186 } 187 188 @VisibleForTesting updateExpansion()189 void updateExpansion() { 190 if (DEBUG_SCALE) Log.v(TAG, "updateExpansion()"); 191 // are we scaling or dragging? 192 float span = mSGD.getCurrentSpan() - mInitialTouchSpan; 193 span *= USE_SPAN ? 1f : 0f; 194 float drag = mSGD.getFocusY() - mInitialTouchFocusY; 195 drag *= USE_DRAG ? 1f : 0f; 196 drag *= mGravity == Gravity.BOTTOM ? -1f : 1f; 197 float pull = Math.abs(drag) + Math.abs(span) + 1f; 198 float hand = drag * Math.abs(drag) / pull + span * Math.abs(span) / pull; 199 float target = hand + mOldHeight; 200 float newHeight = clamp(target); 201 mScaler.setHeight(newHeight); 202 mLastFocusY = mSGD.getFocusY(); 203 mLastSpanY = mSGD.getCurrentSpan(); 204 } 205 clamp(float target)206 private float clamp(float target) { 207 float out = target; 208 out = out < mSmallSize ? mSmallSize : out; 209 out = out > mNaturalHeight ? mNaturalHeight : out; 210 return out; 211 } 212 findView(float x, float y)213 private ExpandableView findView(float x, float y) { 214 ExpandableView v; 215 if (mEventSource != null) { 216 int[] location = new int[2]; 217 mEventSource.getLocationOnScreen(location); 218 x += location[0]; 219 y += location[1]; 220 v = mCallback.getChildAtRawPosition(x, y); 221 } else { 222 v = mCallback.getChildAtPosition(x, y); 223 } 224 return v; 225 } 226 isInside(View v, float x, float y)227 private boolean isInside(View v, float x, float y) { 228 if (DEBUG) Log.d(TAG, "isinside (" + x + ", " + y + ")"); 229 230 if (v == null) { 231 if (DEBUG) Log.d(TAG, "isinside null subject"); 232 return false; 233 } 234 if (mEventSource != null) { 235 int[] location = new int[2]; 236 mEventSource.getLocationOnScreen(location); 237 x += location[0]; 238 y += location[1]; 239 if (DEBUG) Log.d(TAG, " to global (" + x + ", " + y + ")"); 240 } 241 int[] location = new int[2]; 242 v.getLocationOnScreen(location); 243 x -= location[0]; 244 y -= location[1]; 245 if (DEBUG) Log.d(TAG, " to local (" + x + ", " + y + ")"); 246 if (DEBUG) Log.d(TAG, " inside (" + v.getWidth() + ", " + v.getHeight() + ")"); 247 boolean inside = (x > 0f && y > 0f && x < v.getWidth() & y < v.getHeight()); 248 return inside; 249 } 250 setEventSource(View eventSource)251 public void setEventSource(View eventSource) { 252 mEventSource = eventSource; 253 } 254 setGravity(int gravity)255 public void setGravity(int gravity) { 256 mGravity = gravity; 257 } 258 setScrollAdapter(ScrollAdapter adapter)259 public void setScrollAdapter(ScrollAdapter adapter) { 260 mScrollAdapter = adapter; 261 } 262 getTouchSlop(MotionEvent event)263 private float getTouchSlop(MotionEvent event) { 264 // Adjust the touch slop if another gesture may be being performed. 265 return event.getClassification() == MotionEvent.CLASSIFICATION_AMBIGUOUS_GESTURE 266 ? mTouchSlop * mSlopMultiplier 267 : mTouchSlop; 268 } 269 270 @Override onInterceptTouchEvent(MotionEvent ev)271 public boolean onInterceptTouchEvent(MotionEvent ev) { 272 if (!isEnabled()) { 273 return false; 274 } 275 trackVelocity(ev); 276 final int action = ev.getAction(); 277 if (DEBUG_SCALE) Log.d(TAG, "intercept: act=" + MotionEvent.actionToString(action) + 278 " expanding=" + mExpanding + 279 (0 != (mExpansionStyle & BLINDS) ? " (blinds)" : "") + 280 (0 != (mExpansionStyle & PULL) ? " (pull)" : "") + 281 (0 != (mExpansionStyle & STRETCH) ? " (stretch)" : "")); 282 // check for a spread-finger vertical pull gesture 283 mSGD.onTouchEvent(ev); 284 final int x = (int) mSGD.getFocusX(); 285 final int y = (int) mSGD.getFocusY(); 286 287 mInitialTouchFocusY = y; 288 mInitialTouchSpan = mSGD.getCurrentSpan(); 289 mLastFocusY = mInitialTouchFocusY; 290 mLastSpanY = mInitialTouchSpan; 291 if (DEBUG_SCALE) Log.d(TAG, "set initial span: " + mInitialTouchSpan); 292 293 if (mExpanding) { 294 mLastMotionY = ev.getRawY(); 295 maybeRecycleVelocityTracker(ev); 296 return true; 297 } else { 298 if ((action == MotionEvent.ACTION_MOVE) && 0 != (mExpansionStyle & BLINDS)) { 299 // we've begun Venetian blinds style expansion 300 return true; 301 } 302 switch (action & MotionEvent.ACTION_MASK) { 303 case MotionEvent.ACTION_MOVE: { 304 final float xspan = mSGD.getCurrentSpanX(); 305 if (xspan > mPullGestureMinXSpan && 306 xspan > mSGD.getCurrentSpanY() && !mExpanding) { 307 // detect a vertical pulling gesture with fingers somewhat separated 308 if (DEBUG_SCALE) Log.v(TAG, "got pull gesture (xspan=" + xspan + "px)"); 309 startExpanding(mResizedView, PULL); 310 mWatchingForPull = false; 311 } 312 if (mWatchingForPull) { 313 final float yDiff = ev.getRawY() - mInitialTouchY; 314 final float xDiff = ev.getRawX() - mInitialTouchX; 315 if (yDiff > getTouchSlop(ev) && yDiff > Math.abs(xDiff)) { 316 if (DEBUG) Log.v(TAG, "got venetian gesture (dy=" + yDiff + "px)"); 317 mWatchingForPull = false; 318 if (mResizedView != null && !isFullyExpanded(mResizedView)) { 319 if (startExpanding(mResizedView, BLINDS)) { 320 mLastMotionY = ev.getRawY(); 321 mInitialTouchY = ev.getRawY(); 322 mHasPopped = false; 323 } 324 } 325 } 326 } 327 break; 328 } 329 330 case MotionEvent.ACTION_DOWN: 331 mWatchingForPull = mScrollAdapter != null && 332 isInside(mScrollAdapter.getHostView(), x, y) 333 && mScrollAdapter.isScrolledToTop(); 334 mResizedView = findView(x, y); 335 if (mResizedView != null && !mCallback.canChildBeExpanded(mResizedView)) { 336 mResizedView = null; 337 mWatchingForPull = false; 338 } 339 mInitialTouchY = ev.getRawY(); 340 mInitialTouchX = ev.getRawX(); 341 break; 342 343 case MotionEvent.ACTION_CANCEL: 344 case MotionEvent.ACTION_UP: 345 if (DEBUG) Log.d(TAG, "up/cancel"); 346 finishExpanding(ev.getActionMasked() == MotionEvent.ACTION_CANCEL /* forceAbort */, 347 getCurrentVelocity()); 348 clearView(); 349 break; 350 } 351 mLastMotionY = ev.getRawY(); 352 maybeRecycleVelocityTracker(ev); 353 return mExpanding; 354 } 355 } 356 trackVelocity(MotionEvent event)357 private void trackVelocity(MotionEvent event) { 358 int action = event.getActionMasked(); 359 switch(action) { 360 case MotionEvent.ACTION_DOWN: 361 if (mVelocityTracker == null) { 362 mVelocityTracker = VelocityTracker.obtain(); 363 } else { 364 mVelocityTracker.clear(); 365 } 366 mVelocityTracker.addMovement(event); 367 break; 368 case MotionEvent.ACTION_MOVE: 369 if (mVelocityTracker == null) { 370 mVelocityTracker = VelocityTracker.obtain(); 371 } 372 mVelocityTracker.addMovement(event); 373 break; 374 default: 375 break; 376 } 377 } 378 maybeRecycleVelocityTracker(MotionEvent event)379 private void maybeRecycleVelocityTracker(MotionEvent event) { 380 if (mVelocityTracker != null && (event.getActionMasked() == MotionEvent.ACTION_CANCEL 381 || event.getActionMasked() == MotionEvent.ACTION_UP)) { 382 mVelocityTracker.recycle(); 383 mVelocityTracker = null; 384 } 385 } 386 getCurrentVelocity()387 private float getCurrentVelocity() { 388 if (mVelocityTracker != null) { 389 mVelocityTracker.computeCurrentVelocity(1000); 390 return mVelocityTracker.getYVelocity(); 391 } else { 392 return 0f; 393 } 394 } 395 setEnabled(boolean enable)396 public void setEnabled(boolean enable) { 397 mEnabled = enable; 398 } 399 isEnabled()400 private boolean isEnabled() { 401 return mEnabled; 402 } 403 isFullyExpanded(ExpandableView underFocus)404 private boolean isFullyExpanded(ExpandableView underFocus) { 405 return underFocus.getIntrinsicHeight() == underFocus.getMaxContentHeight() 406 && (!underFocus.isSummaryWithChildren() || underFocus.areChildrenExpanded()); 407 } 408 409 @Override onTouchEvent(MotionEvent ev)410 public boolean onTouchEvent(MotionEvent ev) { 411 if (!isEnabled() && !mExpanding) { 412 // In case we're expanding we still want to finish the current motion. 413 return false; 414 } 415 trackVelocity(ev); 416 final int action = ev.getActionMasked(); 417 if (DEBUG_SCALE) Log.d(TAG, "touch: act=" + MotionEvent.actionToString(action) + 418 " expanding=" + mExpanding + 419 (0 != (mExpansionStyle & BLINDS) ? " (blinds)" : "") + 420 (0 != (mExpansionStyle & PULL) ? " (pull)" : "") + 421 (0 != (mExpansionStyle & STRETCH) ? " (stretch)" : "")); 422 423 mSGD.onTouchEvent(ev); 424 final int x = (int) mSGD.getFocusX(); 425 final int y = (int) mSGD.getFocusY(); 426 427 if (mOnlyMovements) { 428 mLastMotionY = ev.getRawY(); 429 return false; 430 } 431 switch (action) { 432 case MotionEvent.ACTION_DOWN: 433 mWatchingForPull = mScrollAdapter != null && 434 isInside(mScrollAdapter.getHostView(), x, y); 435 mResizedView = findView(x, y); 436 mInitialTouchX = ev.getRawX(); 437 mInitialTouchY = ev.getRawY(); 438 break; 439 case MotionEvent.ACTION_MOVE: { 440 if (mWatchingForPull) { 441 final float yDiff = ev.getRawY() - mInitialTouchY; 442 final float xDiff = ev.getRawX() - mInitialTouchX; 443 if (yDiff > getTouchSlop(ev) && yDiff > Math.abs(xDiff)) { 444 if (DEBUG) Log.v(TAG, "got venetian gesture (dy=" + yDiff + "px)"); 445 mWatchingForPull = false; 446 if (mResizedView != null && !isFullyExpanded(mResizedView)) { 447 if (startExpanding(mResizedView, BLINDS)) { 448 mInitialTouchY = ev.getRawY(); 449 mLastMotionY = ev.getRawY(); 450 mHasPopped = false; 451 } 452 } 453 } 454 } 455 if (mExpanding && 0 != (mExpansionStyle & BLINDS)) { 456 final float rawHeight = ev.getRawY() - mLastMotionY + mCurrentHeight; 457 final float newHeight = clamp(rawHeight); 458 boolean isFinished = false; 459 boolean expanded = false; 460 if (rawHeight > mNaturalHeight) { 461 isFinished = true; 462 expanded = true; 463 } 464 if (rawHeight < mSmallSize) { 465 isFinished = true; 466 expanded = false; 467 } 468 469 if (!mHasPopped) { 470 if (mEventSource != null) { 471 mEventSource.performHapticFeedback(HapticFeedbackConstants.VIRTUAL_KEY); 472 } 473 mHasPopped = true; 474 } 475 476 mScaler.setHeight(newHeight); 477 mLastMotionY = ev.getRawY(); 478 if (isFinished) { 479 mCallback.expansionStateChanged(false); 480 } else { 481 mCallback.expansionStateChanged(true); 482 } 483 return true; 484 } 485 486 if (mExpanding) { 487 488 // Gestural expansion is running 489 updateExpansion(); 490 mLastMotionY = ev.getRawY(); 491 return true; 492 } 493 494 break; 495 } 496 497 case MotionEvent.ACTION_POINTER_UP: 498 case MotionEvent.ACTION_POINTER_DOWN: 499 if (DEBUG) Log.d(TAG, "pointer change"); 500 mInitialTouchY += mSGD.getFocusY() - mLastFocusY; 501 mInitialTouchSpan += mSGD.getCurrentSpan() - mLastSpanY; 502 break; 503 504 case MotionEvent.ACTION_UP: 505 case MotionEvent.ACTION_CANCEL: 506 if (DEBUG) Log.d(TAG, "up/cancel"); 507 finishExpanding(!isEnabled() || ev.getActionMasked() == MotionEvent.ACTION_CANCEL, 508 getCurrentVelocity()); 509 clearView(); 510 break; 511 } 512 mLastMotionY = ev.getRawY(); 513 maybeRecycleVelocityTracker(ev); 514 return mResizedView != null; 515 } 516 517 /** 518 * @return True if the view is expandable, false otherwise. 519 */ 520 @VisibleForTesting startExpanding(ExpandableView v, int expandType)521 boolean startExpanding(ExpandableView v, int expandType) { 522 if (!(v instanceof ExpandableNotificationRow)) { 523 return false; 524 } 525 mExpansionStyle = expandType; 526 if (mExpanding && v == mResizedView) { 527 return true; 528 } 529 mExpanding = true; 530 mCallback.expansionStateChanged(true); 531 if (DEBUG) Log.d(TAG, "scale type " + expandType + " beginning on view: " + v); 532 mCallback.setUserLockedChild(v, true); 533 mScaler.setView(v); 534 mOldHeight = mScaler.getHeight(); 535 mCurrentHeight = mOldHeight; 536 boolean canBeExpanded = mCallback.canChildBeExpanded(v); 537 if (canBeExpanded) { 538 if (DEBUG) Log.d(TAG, "working on an expandable child"); 539 mNaturalHeight = mScaler.getNaturalHeight(); 540 mSmallSize = v.getCollapsedHeight(); 541 } else { 542 if (DEBUG) Log.d(TAG, "working on a non-expandable child"); 543 mNaturalHeight = mOldHeight; 544 } 545 if (DEBUG) Log.d(TAG, "got mOldHeight: " + mOldHeight + 546 " mNaturalHeight: " + mNaturalHeight); 547 return true; 548 } 549 550 /** 551 * Finish the current expand motion 552 * @param forceAbort whether the expansion should be forcefully aborted and returned to the old 553 * state 554 * @param velocity the velocity this was expanded/ collapsed with 555 */ 556 @VisibleForTesting finishExpanding(boolean forceAbort, float velocity)557 void finishExpanding(boolean forceAbort, float velocity) { 558 finishExpanding(forceAbort, velocity, true /* allowAnimation */); 559 } 560 561 /** 562 * Finish the current expand motion 563 * @param forceAbort whether the expansion should be forcefully aborted and returned to the old 564 * state 565 * @param velocity the velocity this was expanded/ collapsed with 566 */ finishExpanding(boolean forceAbort, float velocity, boolean allowAnimation)567 private void finishExpanding(boolean forceAbort, float velocity, boolean allowAnimation) { 568 if (!mExpanding) return; 569 570 if (DEBUG) Log.d(TAG, "scale in finishing on view: " + mResizedView); 571 572 float currentHeight = mScaler.getHeight(); 573 final boolean wasClosed = (mOldHeight == mSmallSize); 574 boolean nowExpanded; 575 if (!forceAbort) { 576 if (wasClosed) { 577 nowExpanded = currentHeight > mOldHeight && velocity >= 0; 578 } else { 579 nowExpanded = currentHeight >= mOldHeight || velocity > 0; 580 } 581 nowExpanded |= mNaturalHeight == mSmallSize; 582 } else { 583 nowExpanded = !wasClosed; 584 } 585 if (mScaleAnimation.isRunning()) { 586 mScaleAnimation.cancel(); 587 } 588 mCallback.expansionStateChanged(false); 589 int naturalHeight = mScaler.getNaturalHeight(); 590 float targetHeight = nowExpanded ? naturalHeight : mSmallSize; 591 if (targetHeight != currentHeight && mEnabled && allowAnimation) { 592 mScaleAnimation.setFloatValues(targetHeight); 593 mScaleAnimation.setupStartValues(); 594 final View scaledView = mResizedView; 595 final boolean expand = nowExpanded; 596 mScaleAnimation.addListener(new AnimatorListenerAdapter() { 597 public boolean mCancelled; 598 599 @Override 600 public void onAnimationEnd(Animator animation) { 601 if (!mCancelled) { 602 mCallback.setUserExpandedChild(scaledView, expand); 603 if (!mExpanding) { 604 mScaler.setView(null); 605 } 606 } else { 607 mCallback.setExpansionCancelled(scaledView); 608 } 609 mCallback.setUserLockedChild(scaledView, false); 610 mScaleAnimation.removeListener(this); 611 } 612 613 @Override 614 public void onAnimationCancel(Animator animation) { 615 mCancelled = true; 616 } 617 }); 618 velocity = nowExpanded == velocity >= 0 ? velocity : 0; 619 mFlingAnimationUtils.apply(mScaleAnimation, currentHeight, targetHeight, velocity); 620 mScaleAnimation.start(); 621 } else { 622 if (targetHeight != currentHeight) { 623 mScaler.setHeight(targetHeight); 624 } 625 mCallback.setUserExpandedChild(mResizedView, nowExpanded); 626 mCallback.setUserLockedChild(mResizedView, false); 627 mScaler.setView(null); 628 } 629 630 mExpanding = false; 631 mExpansionStyle = NONE; 632 633 if (DEBUG) Log.d(TAG, "wasClosed is: " + wasClosed); 634 if (DEBUG) Log.d(TAG, "currentHeight is: " + currentHeight); 635 if (DEBUG) Log.d(TAG, "mSmallSize is: " + mSmallSize); 636 if (DEBUG) Log.d(TAG, "targetHeight is: " + targetHeight); 637 if (DEBUG) Log.d(TAG, "scale was finished on view: " + mResizedView); 638 } 639 clearView()640 private void clearView() { 641 mResizedView = null; 642 } 643 644 /** 645 * Use this to abort any pending expansions in progress and force that there will be no 646 * animations. 647 */ cancelImmediately()648 public void cancelImmediately() { 649 cancel(false /* allowAnimation */); 650 } 651 652 /** 653 * Use this to abort any pending expansions in progress. 654 */ cancel()655 public void cancel() { 656 cancel(true /* allowAnimation */); 657 } 658 cancel(boolean allowAnimation)659 private void cancel(boolean allowAnimation) { 660 finishExpanding(true /* forceAbort */, 0f /* velocity */, allowAnimation); 661 clearView(); 662 663 // reset the gesture detector 664 mSGD = new ScaleGestureDetector(mContext, mScaleGestureListener); 665 } 666 667 /** 668 * Change the expansion mode to only observe movements and don't perform any resizing. 669 * This is needed when the expanding is finished and the scroller kicks in, 670 * performing an overscroll motion. We only want to shrink it again when we are not 671 * overscrolled. 672 * 673 * @param onlyMovements Should only movements be observed? 674 */ onlyObserveMovements(boolean onlyMovements)675 public void onlyObserveMovements(boolean onlyMovements) { 676 mOnlyMovements = onlyMovements; 677 } 678 } 679 680