• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2011 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.contacts.quickcontact;
18 
19 import android.animation.Animator;
20 import android.animation.AnimatorListenerAdapter;
21 import android.animation.ObjectAnimator;
22 import android.content.Context;
23 import android.content.res.Resources;
24 import android.graphics.Rect;
25 import android.graphics.drawable.ColorDrawable;
26 import android.graphics.drawable.Drawable;
27 import android.util.AttributeSet;
28 import android.util.Log;
29 import android.view.MotionEvent;
30 import android.view.View;
31 import android.view.animation.AnimationUtils;
32 import android.widget.FrameLayout;
33 import android.widget.PopupWindow;
34 
35 import com.android.contacts.R;
36 import com.android.contacts.test.NeededForReflection;
37 import com.android.contacts.util.SchedulingUtils;
38 
39 /**
40  * Layout containing single child {@link View} which it attempts to center
41  * around {@link #setChildTargetScreen(Rect)}.
42  * <p>
43  * Updates drawable state to be {@link android.R.attr#state_first} when child is
44  * above target, and {@link android.R.attr#state_last} when child is below
45  * target. Also updates {@link Drawable#setLevel(int)} on child
46  * {@link View#getBackground()} to reflect horizontal center of target.
47  * <p>
48  * The reason for this approach is because target {@link Rect} is in screen
49  * coordinates disregarding decor insets; otherwise something like
50  * {@link PopupWindow} might work better.
51  */
52 public class FloatingChildLayout extends FrameLayout {
53     private static final String TAG = "FloatingChildLayout";
54     private int mFixedTopPosition;
55     private View mChild;
56     private Rect mTargetScreen = new Rect();
57     private final int mAnimationDuration;
58 
59     /** The phase of the background dim. This is one of the values of {@link BackgroundPhase}  */
60     private int mBackgroundPhase = BackgroundPhase.BEFORE;
61 
62     private ObjectAnimator mBackgroundAnimator = ObjectAnimator.ofInt(this,
63             "backgroundColorAlpha", 0, DIM_BACKGROUND_ALPHA);
64 
65     private interface BackgroundPhase {
66         public static final int BEFORE = 0;
67         public static final int APPEARING_OR_VISIBLE = 1;
68         public static final int DISAPPEARING_OR_GONE = 3;
69     }
70 
71     /** The phase of the contents window. This is one of the values of {@link ForegroundPhase}  */
72     private int mForegroundPhase = ForegroundPhase.BEFORE;
73 
74     private interface ForegroundPhase {
75         public static final int BEFORE = 0;
76         public static final int APPEARING = 1;
77         public static final int IDLE = 2;
78         public static final int DISAPPEARING = 3;
79         public static final int AFTER = 4;
80     }
81 
82     // Black, 50% alpha as per the system default.
83     private static final int DIM_BACKGROUND_ALPHA = 0x7F;
84 
FloatingChildLayout(Context context, AttributeSet attrs)85     public FloatingChildLayout(Context context, AttributeSet attrs) {
86         super(context, attrs);
87         final Resources resources = getResources();
88         mFixedTopPosition =
89                 resources.getDimensionPixelOffset(R.dimen.quick_contact_top_position);
90         mAnimationDuration = resources.getInteger(android.R.integer.config_shortAnimTime);
91 
92         super.setBackground(new ColorDrawable(0));
93     }
94 
95     @Override
onFinishInflate()96     protected void onFinishInflate() {
97         mChild = findViewById(android.R.id.content);
98         mChild.setDuplicateParentStateEnabled(true);
99 
100         // this will be expanded in showChild()
101         mChild.setScaleX(0.5f);
102         mChild.setScaleY(0.5f);
103         mChild.setAlpha(0.0f);
104     }
105 
getChild()106     public View getChild() {
107         return mChild;
108     }
109 
110     /**
111      * FloatingChildLayout manages its own background, don't set it.
112      */
113     @Override
setBackground(Drawable background)114     public void setBackground(Drawable background) {
115         Log.wtf(TAG, "don't setBackground(), it is managed internally");
116     }
117 
118     /**
119      * Set {@link Rect} in screen coordinates that {@link #getChild()} should be
120      * centered around.
121      */
setChildTargetScreen(Rect targetScreen)122     public void setChildTargetScreen(Rect targetScreen) {
123         mTargetScreen = targetScreen;
124         requestLayout();
125     }
126 
127     /**
128      * Return {@link #mTargetScreen} in local window coordinates, taking any
129      * decor insets into account.
130      */
getTargetInWindow()131     private Rect getTargetInWindow() {
132         final Rect windowScreen = new Rect();
133         getWindowVisibleDisplayFrame(windowScreen);
134 
135         final Rect target = new Rect(mTargetScreen);
136         target.offset(-windowScreen.left, -windowScreen.top);
137         return target;
138     }
139 
140     @Override
onLayout(boolean changed, int left, int top, int right, int bottom)141     protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
142 
143         final View child = mChild;
144         final Rect target = getTargetInWindow();
145 
146         final int childWidth = child.getMeasuredWidth();
147         final int childHeight = child.getMeasuredHeight();
148 
149         if (mFixedTopPosition != -1) {
150             // Horizontally centered, vertically fixed position
151             final int childLeft = (getWidth() - childWidth) / 2;
152             final int childTop = mFixedTopPosition;
153             layoutChild(child, childLeft, childTop);
154         } else {
155             // default is centered horizontally around target...
156             final int childLeft = target.centerX() - (childWidth / 2);
157             // ... and vertically aligned a bit below centered
158             final int childTop = target.centerY() - Math.round(childHeight * 0.35f);
159 
160             // when child is outside bounds, nudge back inside
161             final int clampedChildLeft = clampDimension(childLeft, childWidth, getWidth());
162             final int clampedChildTop = clampDimension(childTop, childHeight, getHeight());
163 
164             layoutChild(child, clampedChildLeft, clampedChildTop);
165         }
166     }
167 
clampDimension(int value, int size, int max)168     private static int clampDimension(int value, int size, int max) {
169         // when larger than bounds, just center
170         if (size > max) {
171             return (max - size) / 2;
172         }
173 
174         // clamp to bounds
175         return Math.min(Math.max(value, 0), max - size);
176     }
177 
layoutChild(View child, int left, int top)178     private static void layoutChild(View child, int left, int top) {
179         child.layout(left, top, left + child.getMeasuredWidth(), top + child.getMeasuredHeight());
180     }
181 
182     @NeededForReflection
setBackgroundColorAlpha(int alpha)183     public void setBackgroundColorAlpha(int alpha) {
184         setBackgroundColor(alpha << 24);
185     }
186 
fadeInBackground()187     public void fadeInBackground() {
188         if (mBackgroundPhase == BackgroundPhase.BEFORE) {
189             mBackgroundPhase = BackgroundPhase.APPEARING_OR_VISIBLE;
190 
191             createChildLayer();
192 
193             SchedulingUtils.doAfterDraw(this, new Runnable() {
194                 @Override
195                 public void run() {
196                     mBackgroundAnimator.setDuration(mAnimationDuration).start();
197                 }
198             });
199         }
200     }
201 
fadeOutBackground()202     public void fadeOutBackground() {
203         if (mBackgroundPhase == BackgroundPhase.APPEARING_OR_VISIBLE) {
204             mBackgroundPhase = BackgroundPhase.DISAPPEARING_OR_GONE;
205             if (mBackgroundAnimator.isRunning()) {
206                 mBackgroundAnimator.reverse();
207             } else {
208                 ObjectAnimator.ofInt(this, "backgroundColorAlpha", DIM_BACKGROUND_ALPHA, 0).
209                         setDuration(mAnimationDuration).start();
210             }
211         }
212     }
213 
isContentFullyVisible()214     public boolean isContentFullyVisible() {
215         return mForegroundPhase == ForegroundPhase.IDLE;
216     }
217 
218     /** Begin animating {@link #getChild()} visible. */
showContent(final Runnable onAnimationEndRunnable)219     public void showContent(final Runnable onAnimationEndRunnable) {
220         if (mForegroundPhase == ForegroundPhase.BEFORE) {
221             mForegroundPhase = ForegroundPhase.APPEARING;
222             animateScale(false, onAnimationEndRunnable);
223         }
224     }
225 
226     /**
227      * Begin animating {@link #getChild()} invisible. Returns false if animation is not valid in
228      * this state
229      */
hideContent(final Runnable onAnimationEndRunnable)230     public boolean hideContent(final Runnable onAnimationEndRunnable) {
231         if (mForegroundPhase == ForegroundPhase.APPEARING ||
232                 mForegroundPhase == ForegroundPhase.IDLE) {
233             mForegroundPhase = ForegroundPhase.DISAPPEARING;
234 
235             createChildLayer();
236 
237             animateScale(true, onAnimationEndRunnable);
238             return true;
239         } else {
240             return false;
241         }
242     }
243 
createChildLayer()244     private void createChildLayer() {
245         mChild.invalidate();
246         mChild.setLayerType(LAYER_TYPE_HARDWARE, null);
247         mChild.buildLayer();
248     }
249 
250     /** Creates the open/close animation */
animateScale( final boolean isExitAnimation, final Runnable onAnimationEndRunnable)251     private void animateScale(
252             final boolean isExitAnimation,
253             final Runnable onAnimationEndRunnable) {
254         mChild.setPivotX(mTargetScreen.centerX() - mChild.getLeft());
255         mChild.setPivotY(mTargetScreen.centerY() - mChild.getTop());
256 
257         final int scaleInterpolator = isExitAnimation
258                 ? android.R.interpolator.accelerate_quint
259                 : android.R.interpolator.decelerate_quint;
260         final float scaleTarget = isExitAnimation ? 0.5f : 1.0f;
261 
262         mChild.animate()
263                 .setDuration(mAnimationDuration)
264                 .setInterpolator(AnimationUtils.loadInterpolator(getContext(), scaleInterpolator))
265                 .scaleX(scaleTarget)
266                 .scaleY(scaleTarget)
267                 .alpha(isExitAnimation ? 0.0f : 1.0f)
268                 .setListener(new AnimatorListenerAdapter() {
269                     @Override
270                     public void onAnimationEnd(Animator animation) {
271                         mChild.setLayerType(LAYER_TYPE_NONE, null);
272                         if (isExitAnimation) {
273                             if (mForegroundPhase == ForegroundPhase.DISAPPEARING) {
274                                 mForegroundPhase = ForegroundPhase.AFTER;
275                                 if (onAnimationEndRunnable != null) onAnimationEndRunnable.run();
276                             }
277                         } else {
278                             if (mForegroundPhase == ForegroundPhase.APPEARING) {
279                                 mForegroundPhase = ForegroundPhase.IDLE;
280                                 if (onAnimationEndRunnable != null) onAnimationEndRunnable.run();
281                             }
282                         }
283                     }
284                 });
285     }
286 
287     private View.OnTouchListener mOutsideTouchListener;
288 
setOnOutsideTouchListener(View.OnTouchListener listener)289     public void setOnOutsideTouchListener(View.OnTouchListener listener) {
290         mOutsideTouchListener = listener;
291     }
292 
293     @Override
onTouchEvent(MotionEvent event)294     public boolean onTouchEvent(MotionEvent event) {
295         // at this point, touch wasn't handled by child view; assume outside
296         if (mOutsideTouchListener != null) {
297             return mOutsideTouchListener.onTouch(this, event);
298         }
299         return false;
300     }
301 }
302