1 /* 2 * Copyright (C) 2018 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 package com.android.systemui.statusbar.notification.stack; 18 19 import static com.android.systemui.statusbar.notification.stack.NotificationSectionsManagerKt.BUCKET_MEDIA_CONTROLS; 20 21 import android.animation.Animator; 22 import android.animation.AnimatorListenerAdapter; 23 import android.animation.ObjectAnimator; 24 import android.animation.PropertyValuesHolder; 25 import android.graphics.Rect; 26 import android.view.View; 27 import android.view.animation.Interpolator; 28 29 import com.android.systemui.Interpolators; 30 import com.android.systemui.statusbar.notification.ShadeViewRefactor; 31 import com.android.systemui.statusbar.notification.row.ExpandableView; 32 33 /** 34 * Represents the bounds of a section of the notification shade and handles animation when the 35 * bounds change. 36 */ 37 public class NotificationSection { 38 private @PriorityBucket int mBucket; 39 private View mOwningView; 40 private Rect mBounds = new Rect(); 41 private Rect mCurrentBounds = new Rect(-1, -1, -1, -1); 42 private Rect mStartAnimationRect = new Rect(); 43 private Rect mEndAnimationRect = new Rect(); 44 private ObjectAnimator mTopAnimator = null; 45 private ObjectAnimator mBottomAnimator = null; 46 private ExpandableView mFirstVisibleChild; 47 private ExpandableView mLastVisibleChild; 48 NotificationSection(View owningView, @PriorityBucket int bucket)49 NotificationSection(View owningView, @PriorityBucket int bucket) { 50 mOwningView = owningView; 51 mBucket = bucket; 52 } 53 cancelAnimators()54 public void cancelAnimators() { 55 if (mBottomAnimator != null) { 56 mBottomAnimator.cancel(); 57 } 58 if (mTopAnimator != null) { 59 mTopAnimator.cancel(); 60 } 61 } 62 getCurrentBounds()63 public Rect getCurrentBounds() { 64 return mCurrentBounds; 65 } 66 getBounds()67 public Rect getBounds() { 68 return mBounds; 69 } 70 didBoundsChange()71 public boolean didBoundsChange() { 72 return !mCurrentBounds.equals(mBounds); 73 } 74 areBoundsAnimating()75 public boolean areBoundsAnimating() { 76 return mBottomAnimator != null || mTopAnimator != null; 77 } 78 79 @PriorityBucket getBucket()80 public int getBucket() { 81 return mBucket; 82 } 83 startBackgroundAnimation(boolean animateTop, boolean animateBottom)84 public void startBackgroundAnimation(boolean animateTop, boolean animateBottom) { 85 // Left and right bounds are always applied immediately. 86 mCurrentBounds.left = mBounds.left; 87 mCurrentBounds.right = mBounds.right; 88 startBottomAnimation(animateBottom); 89 startTopAnimation(animateTop); 90 } 91 92 93 @ShadeViewRefactor(ShadeViewRefactor.RefactorComponent.STATE_RESOLVER) startTopAnimation(boolean animate)94 private void startTopAnimation(boolean animate) { 95 int previousEndValue = mEndAnimationRect.top; 96 int newEndValue = mBounds.top; 97 ObjectAnimator previousAnimator = mTopAnimator; 98 if (previousAnimator != null && previousEndValue == newEndValue) { 99 return; 100 } 101 if (!animate) { 102 // just a local update was performed 103 if (previousAnimator != null) { 104 // we need to increase all animation keyframes of the previous animator by the 105 // relative change to the end value 106 int previousStartValue = mStartAnimationRect.top; 107 PropertyValuesHolder[] values = previousAnimator.getValues(); 108 values[0].setIntValues(previousStartValue, newEndValue); 109 mStartAnimationRect.top = previousStartValue; 110 mEndAnimationRect.top = newEndValue; 111 previousAnimator.setCurrentPlayTime(previousAnimator.getCurrentPlayTime()); 112 return; 113 } else { 114 // no new animation needed, let's just apply the value 115 setBackgroundTop(newEndValue); 116 return; 117 } 118 } 119 if (previousAnimator != null) { 120 previousAnimator.cancel(); 121 } 122 ObjectAnimator animator = ObjectAnimator.ofInt(this, "backgroundTop", 123 mCurrentBounds.top, newEndValue); 124 Interpolator interpolator = Interpolators.FAST_OUT_SLOW_IN; 125 animator.setInterpolator(interpolator); 126 animator.setDuration(StackStateAnimator.ANIMATION_DURATION_STANDARD); 127 // remove the tag when the animation is finished 128 animator.addListener(new AnimatorListenerAdapter() { 129 @Override 130 public void onAnimationEnd(Animator animation) { 131 mStartAnimationRect.top = -1; 132 mEndAnimationRect.top = -1; 133 mTopAnimator = null; 134 } 135 }); 136 animator.start(); 137 mStartAnimationRect.top = mCurrentBounds.top; 138 mEndAnimationRect.top = newEndValue; 139 mTopAnimator = animator; 140 } 141 142 @ShadeViewRefactor(ShadeViewRefactor.RefactorComponent.STATE_RESOLVER) startBottomAnimation(boolean animate)143 private void startBottomAnimation(boolean animate) { 144 int previousStartValue = mStartAnimationRect.bottom; 145 int previousEndValue = mEndAnimationRect.bottom; 146 int newEndValue = mBounds.bottom; 147 ObjectAnimator previousAnimator = mBottomAnimator; 148 if (previousAnimator != null && previousEndValue == newEndValue) { 149 return; 150 } 151 if (!animate) { 152 // just a local update was performed 153 if (previousAnimator != null) { 154 // we need to increase all animation keyframes of the previous animator by the 155 // relative change to the end value 156 PropertyValuesHolder[] values = previousAnimator.getValues(); 157 values[0].setIntValues(previousStartValue, newEndValue); 158 mStartAnimationRect.bottom = previousStartValue; 159 mEndAnimationRect.bottom = newEndValue; 160 previousAnimator.setCurrentPlayTime(previousAnimator.getCurrentPlayTime()); 161 return; 162 } else { 163 // no new animation needed, let's just apply the value 164 setBackgroundBottom(newEndValue); 165 return; 166 } 167 } 168 if (previousAnimator != null) { 169 previousAnimator.cancel(); 170 } 171 ObjectAnimator animator = ObjectAnimator.ofInt(this, "backgroundBottom", 172 mCurrentBounds.bottom, newEndValue); 173 Interpolator interpolator = Interpolators.FAST_OUT_SLOW_IN; 174 animator.setInterpolator(interpolator); 175 animator.setDuration(StackStateAnimator.ANIMATION_DURATION_STANDARD); 176 // remove the tag when the animation is finished 177 animator.addListener(new AnimatorListenerAdapter() { 178 @Override 179 public void onAnimationEnd(Animator animation) { 180 mStartAnimationRect.bottom = -1; 181 mEndAnimationRect.bottom = -1; 182 mBottomAnimator = null; 183 } 184 }); 185 animator.start(); 186 mStartAnimationRect.bottom = mCurrentBounds.bottom; 187 mEndAnimationRect.bottom = newEndValue; 188 mBottomAnimator = animator; 189 } 190 191 @ShadeViewRefactor(ShadeViewRefactor.RefactorComponent.SHADE_VIEW) setBackgroundTop(int top)192 private void setBackgroundTop(int top) { 193 mCurrentBounds.top = top; 194 mOwningView.invalidate(); 195 } 196 197 @ShadeViewRefactor(ShadeViewRefactor.RefactorComponent.SHADE_VIEW) setBackgroundBottom(int bottom)198 private void setBackgroundBottom(int bottom) { 199 mCurrentBounds.bottom = bottom; 200 mOwningView.invalidate(); 201 } 202 getFirstVisibleChild()203 public ExpandableView getFirstVisibleChild() { 204 return mFirstVisibleChild; 205 } 206 getLastVisibleChild()207 public ExpandableView getLastVisibleChild() { 208 return mLastVisibleChild; 209 } 210 setFirstVisibleChild(ExpandableView child)211 public boolean setFirstVisibleChild(ExpandableView child) { 212 boolean changed = mFirstVisibleChild != child; 213 mFirstVisibleChild = child; 214 return changed; 215 } 216 setLastVisibleChild(ExpandableView child)217 public boolean setLastVisibleChild(ExpandableView child) { 218 boolean changed = mLastVisibleChild != child; 219 mLastVisibleChild = child; 220 return changed; 221 } 222 resetCurrentBounds()223 public void resetCurrentBounds() { 224 mCurrentBounds.set(mBounds); 225 } 226 227 /** 228 * Returns true if {@code top} is equal to the top of this section (if not currently animating) 229 * or where the top of this section will be when animation completes. 230 */ isTargetTop(int top)231 public boolean isTargetTop(int top) { 232 return (mTopAnimator == null && mCurrentBounds.top == top) 233 || (mTopAnimator != null && mEndAnimationRect.top == top); 234 } 235 236 /** 237 * Returns true if {@code bottom} is equal to the bottom of this section (if not currently 238 * animating) or where the bottom of this section will be when animation completes. 239 */ isTargetBottom(int bottom)240 public boolean isTargetBottom(int bottom) { 241 return (mBottomAnimator == null && mCurrentBounds.bottom == bottom) 242 || (mBottomAnimator != null && mEndAnimationRect.bottom == bottom); 243 } 244 245 /** 246 * Update the bounds of this section based on it's views 247 * 248 * @param minTopPosition the minimum position that the top needs to have 249 * @param minBottomPosition the minimum position that the bottom needs to have 250 * @return the position of the new bottom 251 */ updateBounds(int minTopPosition, int minBottomPosition, boolean shiftBackgroundWithFirst)252 public int updateBounds(int minTopPosition, int minBottomPosition, 253 boolean shiftBackgroundWithFirst) { 254 int top = minTopPosition; 255 int bottom = minTopPosition; 256 ExpandableView firstView = getFirstVisibleChild(); 257 if (firstView != null) { 258 // Round Y up to avoid seeing the background during animation 259 int finalTranslationY = (int) Math.ceil(ViewState.getFinalTranslationY(firstView)); 260 // TODO: look into the already animating part 261 int newTop; 262 if (isTargetTop(finalTranslationY)) { 263 // we're ending up at the same location as we are now, let's just skip the 264 // animation 265 newTop = finalTranslationY; 266 } else { 267 newTop = (int) Math.ceil(firstView.getTranslationY()); 268 } 269 top = Math.max(newTop, top); 270 if (firstView.showingPulsing()) { 271 // If we're pulsing, the notification can actually go below! 272 bottom = Math.max(bottom, finalTranslationY 273 + ExpandableViewState.getFinalActualHeight(firstView)); 274 if (shiftBackgroundWithFirst) { 275 mBounds.left += Math.max(firstView.getTranslation(), 0); 276 mBounds.right += Math.min(firstView.getTranslation(), 0); 277 } 278 } 279 } 280 top = Math.max(minTopPosition, top); 281 ExpandableView lastView = getLastVisibleChild(); 282 if (lastView != null) { 283 float finalTranslationY = ViewState.getFinalTranslationY(lastView); 284 int finalHeight = ExpandableViewState.getFinalActualHeight(lastView); 285 // Round Y down to avoid seeing the background during animation 286 int finalBottom = (int) Math.floor( 287 finalTranslationY + finalHeight - lastView.getClipBottomAmount()); 288 int newBottom; 289 if (isTargetBottom(finalBottom)) { 290 // we're ending up at the same location as we are now, lets just skip the animation 291 newBottom = finalBottom; 292 } else { 293 newBottom = (int) (lastView.getTranslationY() + lastView.getActualHeight() 294 - lastView.getClipBottomAmount()); 295 // The background can never be lower than the end of the last view 296 minBottomPosition = (int) Math.min( 297 lastView.getTranslationY() + lastView.getActualHeight(), 298 minBottomPosition); 299 } 300 bottom = Math.max(bottom, Math.max(newBottom, minBottomPosition)); 301 } 302 bottom = Math.max(top, bottom); 303 mBounds.top = top; 304 mBounds.bottom = bottom; 305 return bottom; 306 } 307 needsBackground()308 public boolean needsBackground() { 309 return mFirstVisibleChild != null && mBucket != BUCKET_MEDIA_CONTROLS; 310 } 311 } 312