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 package com.android.internal.widget;
17 
18 import android.animation.Animator;
19 import android.animation.AnimatorSet;
20 import android.animation.ObjectAnimator;
21 import android.animation.TimeInterpolator;
22 import android.compat.annotation.UnsupportedAppUsage;
23 import android.content.Context;
24 import android.content.res.Configuration;
25 import android.content.res.TypedArray;
26 import android.os.Build;
27 import android.util.AttributeSet;
28 import android.util.TypedValue;
29 import android.view.ContextThemeWrapper;
30 import android.view.MotionEvent;
31 import android.view.View;
32 import android.view.ViewGroup;
33 import android.view.animation.DecelerateInterpolator;
34 import android.widget.ActionMenuPresenter;
35 import android.widget.ActionMenuView;
36 
37 import com.android.internal.R;
38 
39 public abstract class AbsActionBarView extends ViewGroup {
40     private static final TimeInterpolator sAlphaInterpolator = new DecelerateInterpolator();
41 
42     private static final int FADE_DURATION = 200;
43 
44     protected final VisibilityAnimListener mVisAnimListener = new VisibilityAnimListener();
45 
46     /** Context against which to inflate popup menus. */
47     protected final Context mPopupContext;
48 
49     protected ActionMenuView mMenuView;
50     protected ActionMenuPresenter mActionMenuPresenter;
51     protected ViewGroup mSplitView;
52     protected boolean mSplitActionBar;
53     protected boolean mSplitWhenNarrow;
54     protected int mContentHeight;
55 
56     protected Animator mVisibilityAnim;
57 
58     private boolean mEatingTouch;
59     private boolean mEatingHover;
60 
AbsActionBarView(Context context)61     public AbsActionBarView(Context context) {
62         this(context, null);
63     }
64 
AbsActionBarView(Context context, AttributeSet attrs)65     public AbsActionBarView(Context context, AttributeSet attrs) {
66         this(context, attrs, 0);
67     }
68 
AbsActionBarView(Context context, AttributeSet attrs, int defStyleAttr)69     public AbsActionBarView(Context context, AttributeSet attrs, int defStyleAttr) {
70         this(context, attrs, defStyleAttr, 0);
71     }
72 
AbsActionBarView( Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes)73     public AbsActionBarView(
74             Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
75         super(context, attrs, defStyleAttr, defStyleRes);
76 
77         final TypedValue tv = new TypedValue();
78         if (context.getTheme().resolveAttribute(R.attr.actionBarPopupTheme, tv, true)
79                 && tv.resourceId != 0) {
80             mPopupContext = new ContextThemeWrapper(context, tv.resourceId);
81         } else {
82             mPopupContext = context;
83         }
84     }
85 
86     @Override
onConfigurationChanged(Configuration newConfig)87     protected void onConfigurationChanged(Configuration newConfig) {
88         super.onConfigurationChanged(newConfig);
89 
90         // Action bar can change size on configuration changes.
91         // Reread the desired height from the theme-specified style.
92         TypedArray a = getContext().obtainStyledAttributes(null, R.styleable.ActionBar,
93                 com.android.internal.R.attr.actionBarStyle, 0);
94         setContentHeight(a.getLayoutDimension(R.styleable.ActionBar_height, 0));
95         a.recycle();
96         if (mSplitWhenNarrow) {
97             setSplitToolbar(getContext().getResources().getBoolean(
98                     com.android.internal.R.bool.split_action_bar_is_narrow));
99         }
100         if (mActionMenuPresenter != null) {
101             mActionMenuPresenter.onConfigurationChanged(newConfig);
102         }
103     }
104 
105     @Override
onTouchEvent(MotionEvent ev)106     public boolean onTouchEvent(MotionEvent ev) {
107         // ActionBarViews always eat touch events, but should still respect the touch event dispatch
108         // contract. If the normal View implementation doesn't want the events, we'll just silently
109         // eat the rest of the gesture without reporting the events to the default implementation
110         // since that's what it expects.
111 
112         final int action = ev.getActionMasked();
113         if (action == MotionEvent.ACTION_DOWN) {
114             mEatingTouch = false;
115         }
116 
117         if (!mEatingTouch) {
118             final boolean handled = super.onTouchEvent(ev);
119             if (action == MotionEvent.ACTION_DOWN && !handled) {
120                 mEatingTouch = true;
121             }
122         }
123 
124         if (action == MotionEvent.ACTION_UP || action == MotionEvent.ACTION_CANCEL) {
125             mEatingTouch = false;
126         }
127 
128         return true;
129     }
130 
131     @Override
onHoverEvent(MotionEvent ev)132     public boolean onHoverEvent(MotionEvent ev) {
133         // Same deal as onTouchEvent() above. Eat all hover events, but still
134         // respect the touch event dispatch contract.
135 
136         final int action = ev.getActionMasked();
137         if (action == MotionEvent.ACTION_HOVER_ENTER) {
138             mEatingHover = false;
139         }
140 
141         if (!mEatingHover) {
142             final boolean handled = super.onHoverEvent(ev);
143             if (action == MotionEvent.ACTION_HOVER_ENTER && !handled) {
144                 mEatingHover = true;
145             }
146         }
147 
148         if (action == MotionEvent.ACTION_HOVER_EXIT
149                 || action == MotionEvent.ACTION_CANCEL) {
150             mEatingHover = false;
151         }
152 
153         return true;
154     }
155 
156     /**
157      * Sets whether the bar should be split right now, no questions asked.
158      * @param split true if the bar should split
159      */
setSplitToolbar(boolean split)160     public void setSplitToolbar(boolean split) {
161         mSplitActionBar = split;
162     }
163 
164     /**
165      * Sets whether the bar should split if we enter a narrow screen configuration.
166      * @param splitWhenNarrow true if the bar should check to split after a config change
167      */
setSplitWhenNarrow(boolean splitWhenNarrow)168     public void setSplitWhenNarrow(boolean splitWhenNarrow) {
169         mSplitWhenNarrow = splitWhenNarrow;
170     }
171 
setContentHeight(int height)172     public void setContentHeight(int height) {
173         mContentHeight = height;
174         requestLayout();
175     }
176 
getContentHeight()177     public int getContentHeight() {
178         return mContentHeight;
179     }
180 
setSplitView(ViewGroup splitView)181     public void setSplitView(ViewGroup splitView) {
182         mSplitView = splitView;
183     }
184 
185     /**
186      * @return Current visibility or if animating, the visibility being animated to.
187      */
getAnimatedVisibility()188     public int getAnimatedVisibility() {
189         if (mVisibilityAnim != null) {
190             return mVisAnimListener.mFinalVisibility;
191         }
192         return getVisibility();
193     }
194 
setupAnimatorToVisibility(int visibility, long duration)195     public Animator setupAnimatorToVisibility(int visibility, long duration) {
196         if (mVisibilityAnim != null) {
197             mVisibilityAnim.cancel();
198         }
199 
200         if (visibility == VISIBLE) {
201             if (getVisibility() != VISIBLE) {
202                 setAlpha(0);
203                 if (mSplitView != null && mMenuView != null) {
204                     mMenuView.setAlpha(0);
205                 }
206             }
207             ObjectAnimator anim = ObjectAnimator.ofFloat(this, View.ALPHA, 1);
208             anim.setDuration(duration);
209             anim.setInterpolator(sAlphaInterpolator);
210             if (mSplitView != null && mMenuView != null) {
211                 AnimatorSet set = new AnimatorSet();
212                 ObjectAnimator splitAnim = ObjectAnimator.ofFloat(mMenuView, View.ALPHA, 1);
213                 splitAnim.setDuration(duration);
214                 set.addListener(mVisAnimListener.withFinalVisibility(visibility));
215                 set.play(anim).with(splitAnim);
216                 return set;
217             } else {
218                 anim.addListener(mVisAnimListener.withFinalVisibility(visibility));
219                 return anim;
220             }
221         } else {
222             ObjectAnimator anim = ObjectAnimator.ofFloat(this, View.ALPHA, 0);
223             anim.setDuration(duration);
224             anim.setInterpolator(sAlphaInterpolator);
225             if (mSplitView != null && mMenuView != null) {
226                 AnimatorSet set = new AnimatorSet();
227                 ObjectAnimator splitAnim = ObjectAnimator.ofFloat(mMenuView, View.ALPHA, 0);
228                 splitAnim.setDuration(duration);
229                 set.addListener(mVisAnimListener.withFinalVisibility(visibility));
230                 set.play(anim).with(splitAnim);
231                 return set;
232             } else {
233                 anim.addListener(mVisAnimListener.withFinalVisibility(visibility));
234                 return anim;
235             }
236         }
237     }
238 
animateToVisibility(int visibility)239     public void animateToVisibility(int visibility) {
240         Animator anim = setupAnimatorToVisibility(visibility, FADE_DURATION);
241         anim.start();
242     }
243 
244     @Override
setVisibility(int visibility)245     public void setVisibility(int visibility) {
246         if (visibility != getVisibility()) {
247             if (mVisibilityAnim != null) {
248                 mVisibilityAnim.end();
249             }
250             super.setVisibility(visibility);
251         }
252     }
253 
showOverflowMenu()254     public boolean showOverflowMenu() {
255         if (mActionMenuPresenter != null) {
256             return mActionMenuPresenter.showOverflowMenu();
257         }
258         return false;
259     }
260 
postShowOverflowMenu()261     public void postShowOverflowMenu() {
262         post(new Runnable() {
263             public void run() {
264                 showOverflowMenu();
265             }
266         });
267     }
268 
hideOverflowMenu()269     public boolean hideOverflowMenu() {
270         if (mActionMenuPresenter != null) {
271             return mActionMenuPresenter.hideOverflowMenu();
272         }
273         return false;
274     }
275 
isOverflowMenuShowing()276     public boolean isOverflowMenuShowing() {
277         if (mActionMenuPresenter != null) {
278             return mActionMenuPresenter.isOverflowMenuShowing();
279         }
280         return false;
281     }
282 
isOverflowMenuShowPending()283     public boolean isOverflowMenuShowPending() {
284         if (mActionMenuPresenter != null) {
285             return mActionMenuPresenter.isOverflowMenuShowPending();
286         }
287         return false;
288     }
289 
isOverflowReserved()290     public boolean isOverflowReserved() {
291         return mActionMenuPresenter != null && mActionMenuPresenter.isOverflowReserved();
292     }
293 
canShowOverflowMenu()294     public boolean canShowOverflowMenu() {
295         return isOverflowReserved() && getVisibility() == VISIBLE;
296     }
297 
298     @UnsupportedAppUsage(maxTargetSdk = Build.VERSION_CODES.R, trackingBug = 170729553)
dismissPopupMenus()299     public void dismissPopupMenus() {
300         if (mActionMenuPresenter != null) {
301             mActionMenuPresenter.dismissPopupMenus();
302         }
303     }
304 
measureChildView(View child, int availableWidth, int childSpecHeight, int spacing)305     protected int measureChildView(View child, int availableWidth, int childSpecHeight,
306             int spacing) {
307         child.measure(MeasureSpec.makeMeasureSpec(availableWidth, MeasureSpec.AT_MOST),
308                 childSpecHeight);
309 
310         availableWidth -= child.getMeasuredWidth();
311         availableWidth -= spacing;
312 
313         return Math.max(0, availableWidth);
314     }
315 
next(int x, int val, boolean isRtl)316     static protected int next(int x, int val, boolean isRtl) {
317         return isRtl ? x - val : x + val;
318     }
319 
positionChild(View child, int x, int y, int contentHeight, boolean reverse)320     protected int positionChild(View child, int x, int y, int contentHeight, boolean reverse) {
321         int childWidth = child.getMeasuredWidth();
322         int childHeight = child.getMeasuredHeight();
323         int childTop = y + (contentHeight - childHeight) / 2;
324 
325         if (reverse) {
326             child.layout(x - childWidth, childTop, x, childTop + childHeight);
327         } else {
328             child.layout(x, childTop, x + childWidth, childTop + childHeight);
329         }
330 
331         return  (reverse ? -childWidth : childWidth);
332     }
333 
334     protected class VisibilityAnimListener implements Animator.AnimatorListener {
335         private boolean mCanceled = false;
336         int mFinalVisibility;
337 
withFinalVisibility(int visibility)338         public VisibilityAnimListener withFinalVisibility(int visibility) {
339             mFinalVisibility = visibility;
340             return this;
341         }
342 
343         @Override
onAnimationStart(Animator animation)344         public void onAnimationStart(Animator animation) {
345             setVisibility(VISIBLE);
346             mVisibilityAnim = animation;
347             mCanceled = false;
348         }
349 
350         @Override
onAnimationEnd(Animator animation)351         public void onAnimationEnd(Animator animation) {
352             if (mCanceled) return;
353 
354             mVisibilityAnim = null;
355             setVisibility(mFinalVisibility);
356             if (mSplitView != null && mMenuView != null) {
357                 mMenuView.setVisibility(mFinalVisibility);
358             }
359         }
360 
361         @Override
onAnimationCancel(Animator animation)362         public void onAnimationCancel(Animator animation) {
363             mCanceled = true;
364         }
365 
366         @Override
onAnimationRepeat(Animator animation)367         public void onAnimationRepeat(Animator animation) {
368         }
369     }
370 }
371