1 /* 2 * Copyright (C) 2014 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.row; 18 19 import android.animation.AnimatorListenerAdapter; 20 import android.content.Context; 21 import android.util.AttributeSet; 22 import android.view.View; 23 import android.view.animation.Interpolator; 24 25 import com.android.internal.annotations.VisibleForTesting; 26 import com.android.systemui.Interpolators; 27 28 /** 29 * A common base class for all views in the notification stack scroller which don't have a 30 * background. 31 */ 32 public abstract class StackScrollerDecorView extends ExpandableView { 33 34 protected View mContent; 35 protected View mSecondaryView; 36 private boolean mIsVisible = true; 37 private boolean mContentVisible = true; 38 private boolean mIsSecondaryVisible = true; 39 private int mDuration = 260; 40 private boolean mContentAnimating; 41 private final Runnable mContentVisibilityEndRunnable = () -> { 42 mContentAnimating = false; 43 if (getVisibility() != View.GONE && !mIsVisible) { 44 setVisibility(GONE); 45 setWillBeGone(false); 46 notifyHeightChanged(false /* needsAnimation */); 47 } 48 }; 49 50 private boolean mSecondaryAnimating = false; 51 private final Runnable mSecondaryVisibilityEndRunnable = () -> { 52 mSecondaryAnimating = false; 53 // If we were on screen, become GONE to avoid touches 54 if (mSecondaryView == null) return; 55 if (getVisibility() != View.GONE 56 && mSecondaryView.getVisibility() != View.GONE 57 && !mIsSecondaryVisible) { 58 mSecondaryView.setVisibility(View.GONE); 59 } 60 }; 61 StackScrollerDecorView(Context context, AttributeSet attrs)62 public StackScrollerDecorView(Context context, AttributeSet attrs) { 63 super(context, attrs); 64 setClipChildren(false); 65 } 66 67 @Override onFinishInflate()68 protected void onFinishInflate() { 69 super.onFinishInflate(); 70 mContent = findContentView(); 71 mSecondaryView = findSecondaryView(); 72 setVisible(false /* nowVisible */, false /* animate */); 73 setSecondaryVisible(false /* nowVisible */, false /* animate */); 74 } 75 76 @Override onLayout(boolean changed, int left, int top, int right, int bottom)77 protected void onLayout(boolean changed, int left, int top, int right, int bottom) { 78 super.onLayout(changed, left, top, right, bottom); 79 setOutlineProvider(null); 80 } 81 82 @Override isTransparent()83 public boolean isTransparent() { 84 return true; 85 } 86 87 /** 88 * Set the content of this view to be visible in an animated way. 89 * 90 * @param contentVisible True if the content should be visible or false if it should be hidden. 91 */ setContentVisible(boolean contentVisible)92 public void setContentVisible(boolean contentVisible) { 93 setContentVisible(contentVisible, true /* animate */); 94 } 95 /** 96 * Set the content of this view to be visible. 97 * @param contentVisible True if the content should be visible or false if it should be hidden. 98 * @param animate Should an animation be performed. 99 */ setContentVisible(boolean contentVisible, boolean animate)100 private void setContentVisible(boolean contentVisible, boolean animate) { 101 if (mContentVisible != contentVisible) { 102 mContentAnimating = animate; 103 mContentVisible = contentVisible; 104 setViewVisible(mContent, contentVisible, animate, mContentVisibilityEndRunnable); 105 } 106 107 if (!mContentAnimating) { 108 mContentVisibilityEndRunnable.run(); 109 } 110 } 111 isContentVisible()112 public boolean isContentVisible() { 113 return mContentVisible; 114 } 115 116 /** 117 * Make this view visible. If {@code false} is passed, the view will fade out it's content 118 * and set the view Visibility to GONE. If only the content should be changed 119 * {@link #setContentVisible(boolean)} can be used. 120 * 121 * @param nowVisible should the view be visible 122 * @param animate should the change be animated. 123 */ setVisible(boolean nowVisible, boolean animate)124 public void setVisible(boolean nowVisible, boolean animate) { 125 if (mIsVisible != nowVisible) { 126 mIsVisible = nowVisible; 127 if (animate) { 128 if (nowVisible) { 129 setVisibility(VISIBLE); 130 setWillBeGone(false); 131 notifyHeightChanged(false /* needsAnimation */); 132 } else { 133 setWillBeGone(true); 134 } 135 setContentVisible(nowVisible, true /* animate */); 136 } else { 137 setVisibility(nowVisible ? VISIBLE : GONE); 138 setContentVisible(nowVisible, false /* animate */); 139 setWillBeGone(false); 140 notifyHeightChanged(false /* needsAnimation */); 141 } 142 } 143 } 144 145 /** 146 * Set the secondary view of this layout to visible. 147 * 148 * @param nowVisible should the secondary view be visible 149 * @param animate should the change be animated 150 */ setSecondaryVisible(boolean nowVisible, boolean animate)151 public void setSecondaryVisible(boolean nowVisible, boolean animate) { 152 if (mIsSecondaryVisible != nowVisible) { 153 mSecondaryAnimating = animate; 154 mIsSecondaryVisible = nowVisible; 155 setViewVisible(mSecondaryView, nowVisible, animate, mSecondaryVisibilityEndRunnable); 156 } 157 158 if (!mSecondaryAnimating) { 159 mSecondaryVisibilityEndRunnable.run(); 160 } 161 } 162 163 @VisibleForTesting isSecondaryVisible()164 boolean isSecondaryVisible() { 165 return mIsSecondaryVisible; 166 } 167 168 /** 169 * Is this view visible. If a view is currently animating to gone, it will 170 * return {@code false}. 171 */ isVisible()172 public boolean isVisible() { 173 return mIsVisible; 174 } 175 setDuration(int duration)176 void setDuration(int duration) { 177 mDuration = duration; 178 } 179 180 /** 181 * Animate a view to a new visibility. 182 * @param view Target view, maybe content view or dismiss view. 183 * @param nowVisible Should it now be visible. 184 * @param animate Should this be done in an animated way. 185 * @param endRunnable A runnable that is run when the animation is done. 186 */ setViewVisible(View view, boolean nowVisible, boolean animate, Runnable endRunnable)187 private void setViewVisible(View view, boolean nowVisible, 188 boolean animate, Runnable endRunnable) { 189 if (view == null) { 190 return; 191 } 192 193 // Make sure we're visible so animations work 194 if (view.getVisibility() != View.VISIBLE) { 195 view.setVisibility(View.VISIBLE); 196 } 197 198 // cancel any previous animations 199 view.animate().cancel(); 200 float endValue = nowVisible ? 1.0f : 0.0f; 201 if (!animate) { 202 view.setAlpha(endValue); 203 if (endRunnable != null) { 204 endRunnable.run(); 205 } 206 return; 207 } 208 209 // Animate the view alpha 210 Interpolator interpolator = nowVisible ? Interpolators.ALPHA_IN : Interpolators.ALPHA_OUT; 211 view.animate() 212 .alpha(endValue) 213 .setInterpolator(interpolator) 214 .setDuration(mDuration) 215 .withEndAction(endRunnable); 216 } 217 218 @Override performRemoveAnimation(long duration, long delay, float translationDirection, boolean isHeadsUpAnimation, float endLocation, Runnable onFinishedRunnable, AnimatorListenerAdapter animationListener)219 public long performRemoveAnimation(long duration, long delay, 220 float translationDirection, boolean isHeadsUpAnimation, float endLocation, 221 Runnable onFinishedRunnable, 222 AnimatorListenerAdapter animationListener) { 223 // TODO: Use duration 224 setContentVisible(false); 225 return 0; 226 } 227 228 @Override performAddAnimation(long delay, long duration, boolean isHeadsUpAppear)229 public void performAddAnimation(long delay, long duration, boolean isHeadsUpAppear) { 230 // TODO: use delay and duration 231 setContentVisible(true); 232 } 233 234 @Override needsClippingToShelf()235 public boolean needsClippingToShelf() { 236 return false; 237 } 238 239 @Override hasOverlappingRendering()240 public boolean hasOverlappingRendering() { 241 return false; 242 } 243 findContentView()244 protected abstract View findContentView(); 245 246 /** 247 * Returns a view that might not always appear while the main content view is still visible. 248 */ findSecondaryView()249 protected abstract View findSecondaryView(); 250 } 251