1 /*
2  * Copyright (C) 2006 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 android.widget;
18 
19 import android.annotation.AttrRes;
20 import android.annotation.NonNull;
21 import android.annotation.Nullable;
22 import android.annotation.StyleRes;
23 import android.content.Context;
24 import android.content.res.TypedArray;
25 import android.graphics.Rect;
26 import android.graphics.drawable.Drawable;
27 import android.util.AttributeSet;
28 import android.view.Gravity;
29 import android.view.View;
30 import android.view.ViewDebug;
31 import android.view.ViewGroup;
32 import android.view.ViewHierarchyEncoder;
33 import android.widget.RemoteViews.RemoteView;
34 
35 import com.android.internal.R;
36 
37 import java.util.ArrayList;
38 
39 /**
40  * FrameLayout is designed to block out an area on the screen to display
41  * a single item. Generally, FrameLayout should be used to hold a single child view, because it can
42  * be difficult to organize child views in a way that's scalable to different screen sizes without
43  * the children overlapping each other. You can, however, add multiple children to a FrameLayout
44  * and control their position within the FrameLayout by assigning gravity to each child, using the
45  * <a href="FrameLayout.LayoutParams.html#attr_android:layout_gravity">{@code
46  * android:layout_gravity}</a> attribute.
47  * <p>Child views are drawn in a stack, with the most recently added child on top.
48  * The size of the FrameLayout is the size of its largest child (plus padding), visible
49  * or not (if the FrameLayout's parent permits). Views that are {@link android.view.View#GONE} are
50  * used for sizing
51  * only if {@link #setMeasureAllChildren(boolean) setConsiderGoneChildrenWhenMeasuring()}
52  * is set to true.
53  *
54  * @attr ref android.R.styleable#FrameLayout_measureAllChildren
55  */
56 @RemoteView
57 public class FrameLayout extends ViewGroup {
58     private static final int DEFAULT_CHILD_GRAVITY = Gravity.TOP | Gravity.START;
59 
60     @ViewDebug.ExportedProperty(category = "measurement")
61     boolean mMeasureAllChildren = false;
62 
63     @ViewDebug.ExportedProperty(category = "padding")
64     private int mForegroundPaddingLeft = 0;
65 
66     @ViewDebug.ExportedProperty(category = "padding")
67     private int mForegroundPaddingTop = 0;
68 
69     @ViewDebug.ExportedProperty(category = "padding")
70     private int mForegroundPaddingRight = 0;
71 
72     @ViewDebug.ExportedProperty(category = "padding")
73     private int mForegroundPaddingBottom = 0;
74 
75     private final ArrayList<View> mMatchParentChildren = new ArrayList<>(1);
76 
FrameLayout(@onNull Context context)77     public FrameLayout(@NonNull Context context) {
78         super(context);
79     }
80 
FrameLayout(@onNull Context context, @Nullable AttributeSet attrs)81     public FrameLayout(@NonNull Context context, @Nullable AttributeSet attrs) {
82         this(context, attrs, 0);
83     }
84 
FrameLayout(@onNull Context context, @Nullable AttributeSet attrs, @AttrRes int defStyleAttr)85     public FrameLayout(@NonNull Context context, @Nullable AttributeSet attrs,
86             @AttrRes int defStyleAttr) {
87         this(context, attrs, defStyleAttr, 0);
88     }
89 
FrameLayout(@onNull Context context, @Nullable AttributeSet attrs, @AttrRes int defStyleAttr, @StyleRes int defStyleRes)90     public FrameLayout(@NonNull Context context, @Nullable AttributeSet attrs,
91             @AttrRes int defStyleAttr, @StyleRes int defStyleRes) {
92         super(context, attrs, defStyleAttr, defStyleRes);
93 
94         final TypedArray a = context.obtainStyledAttributes(
95                 attrs, R.styleable.FrameLayout, defStyleAttr, defStyleRes);
96 
97         if (a.getBoolean(R.styleable.FrameLayout_measureAllChildren, false)) {
98             setMeasureAllChildren(true);
99         }
100 
101         a.recycle();
102     }
103 
104     /**
105      * Describes how the foreground is positioned. Defaults to START and TOP.
106      *
107      * @param foregroundGravity See {@link android.view.Gravity}
108      *
109      * @see #getForegroundGravity()
110      *
111      * @attr ref android.R.styleable#View_foregroundGravity
112      */
113     @android.view.RemotableViewMethod
setForegroundGravity(int foregroundGravity)114     public void setForegroundGravity(int foregroundGravity) {
115         if (getForegroundGravity() != foregroundGravity) {
116             super.setForegroundGravity(foregroundGravity);
117 
118             // calling get* again here because the set above may apply default constraints
119             final Drawable foreground = getForeground();
120             if (getForegroundGravity() == Gravity.FILL && foreground != null) {
121                 Rect padding = new Rect();
122                 if (foreground.getPadding(padding)) {
123                     mForegroundPaddingLeft = padding.left;
124                     mForegroundPaddingTop = padding.top;
125                     mForegroundPaddingRight = padding.right;
126                     mForegroundPaddingBottom = padding.bottom;
127                 }
128             } else {
129                 mForegroundPaddingLeft = 0;
130                 mForegroundPaddingTop = 0;
131                 mForegroundPaddingRight = 0;
132                 mForegroundPaddingBottom = 0;
133             }
134 
135             requestLayout();
136         }
137     }
138 
139     /**
140      * Returns a set of layout parameters with a width of
141      * {@link android.view.ViewGroup.LayoutParams#MATCH_PARENT},
142      * and a height of {@link android.view.ViewGroup.LayoutParams#MATCH_PARENT}.
143      */
144     @Override
generateDefaultLayoutParams()145     protected LayoutParams generateDefaultLayoutParams() {
146         return new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT);
147     }
148 
getPaddingLeftWithForeground()149     int getPaddingLeftWithForeground() {
150         return isForegroundInsidePadding() ? Math.max(mPaddingLeft, mForegroundPaddingLeft) :
151             mPaddingLeft + mForegroundPaddingLeft;
152     }
153 
getPaddingRightWithForeground()154     int getPaddingRightWithForeground() {
155         return isForegroundInsidePadding() ? Math.max(mPaddingRight, mForegroundPaddingRight) :
156             mPaddingRight + mForegroundPaddingRight;
157     }
158 
getPaddingTopWithForeground()159     private int getPaddingTopWithForeground() {
160         return isForegroundInsidePadding() ? Math.max(mPaddingTop, mForegroundPaddingTop) :
161             mPaddingTop + mForegroundPaddingTop;
162     }
163 
getPaddingBottomWithForeground()164     private int getPaddingBottomWithForeground() {
165         return isForegroundInsidePadding() ? Math.max(mPaddingBottom, mForegroundPaddingBottom) :
166             mPaddingBottom + mForegroundPaddingBottom;
167     }
168 
169     @Override
onMeasure(int widthMeasureSpec, int heightMeasureSpec)170     protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
171         int count = getChildCount();
172 
173         final boolean measureMatchParentChildren =
174                 MeasureSpec.getMode(widthMeasureSpec) != MeasureSpec.EXACTLY ||
175                 MeasureSpec.getMode(heightMeasureSpec) != MeasureSpec.EXACTLY;
176         mMatchParentChildren.clear();
177 
178         int maxHeight = 0;
179         int maxWidth = 0;
180         int childState = 0;
181 
182         for (int i = 0; i < count; i++) {
183             final View child = getChildAt(i);
184             if (mMeasureAllChildren || child.getVisibility() != GONE) {
185                 measureChildWithMargins(child, widthMeasureSpec, 0, heightMeasureSpec, 0);
186                 final LayoutParams lp = (LayoutParams) child.getLayoutParams();
187                 maxWidth = Math.max(maxWidth,
188                         child.getMeasuredWidth() + lp.leftMargin + lp.rightMargin);
189                 maxHeight = Math.max(maxHeight,
190                         child.getMeasuredHeight() + lp.topMargin + lp.bottomMargin);
191                 childState = combineMeasuredStates(childState, child.getMeasuredState());
192                 if (measureMatchParentChildren) {
193                     if (lp.width == LayoutParams.MATCH_PARENT ||
194                             lp.height == LayoutParams.MATCH_PARENT) {
195                         mMatchParentChildren.add(child);
196                     }
197                 }
198             }
199         }
200 
201         // Account for padding too
202         maxWidth += getPaddingLeftWithForeground() + getPaddingRightWithForeground();
203         maxHeight += getPaddingTopWithForeground() + getPaddingBottomWithForeground();
204 
205         // Check against our minimum height and width
206         maxHeight = Math.max(maxHeight, getSuggestedMinimumHeight());
207         maxWidth = Math.max(maxWidth, getSuggestedMinimumWidth());
208 
209         // Check against our foreground's minimum height and width
210         final Drawable drawable = getForeground();
211         if (drawable != null) {
212             maxHeight = Math.max(maxHeight, drawable.getMinimumHeight());
213             maxWidth = Math.max(maxWidth, drawable.getMinimumWidth());
214         }
215 
216         setMeasuredDimension(resolveSizeAndState(maxWidth, widthMeasureSpec, childState),
217                 resolveSizeAndState(maxHeight, heightMeasureSpec,
218                         childState << MEASURED_HEIGHT_STATE_SHIFT));
219 
220         count = mMatchParentChildren.size();
221         if (count > 1) {
222             for (int i = 0; i < count; i++) {
223                 final View child = mMatchParentChildren.get(i);
224                 final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
225 
226                 final int childWidthMeasureSpec;
227                 if (lp.width == LayoutParams.MATCH_PARENT) {
228                     final int width = Math.max(0, getMeasuredWidth()
229                             - getPaddingLeftWithForeground() - getPaddingRightWithForeground()
230                             - lp.leftMargin - lp.rightMargin);
231                     childWidthMeasureSpec = MeasureSpec.makeMeasureSpec(
232                             width, MeasureSpec.EXACTLY);
233                 } else {
234                     childWidthMeasureSpec = getChildMeasureSpec(widthMeasureSpec,
235                             getPaddingLeftWithForeground() + getPaddingRightWithForeground() +
236                             lp.leftMargin + lp.rightMargin,
237                             lp.width);
238                 }
239 
240                 final int childHeightMeasureSpec;
241                 if (lp.height == LayoutParams.MATCH_PARENT) {
242                     final int height = Math.max(0, getMeasuredHeight()
243                             - getPaddingTopWithForeground() - getPaddingBottomWithForeground()
244                             - lp.topMargin - lp.bottomMargin);
245                     childHeightMeasureSpec = MeasureSpec.makeMeasureSpec(
246                             height, MeasureSpec.EXACTLY);
247                 } else {
248                     childHeightMeasureSpec = getChildMeasureSpec(heightMeasureSpec,
249                             getPaddingTopWithForeground() + getPaddingBottomWithForeground() +
250                             lp.topMargin + lp.bottomMargin,
251                             lp.height);
252                 }
253 
254                 child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
255             }
256         }
257     }
258 
259     @Override
onLayout(boolean changed, int left, int top, int right, int bottom)260     protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
261         layoutChildren(left, top, right, bottom, false /* no force left gravity */);
262     }
263 
layoutChildren(int left, int top, int right, int bottom, boolean forceLeftGravity)264     void layoutChildren(int left, int top, int right, int bottom, boolean forceLeftGravity) {
265         final int count = getChildCount();
266 
267         final int parentLeft = getPaddingLeftWithForeground();
268         final int parentRight = right - left - getPaddingRightWithForeground();
269 
270         final int parentTop = getPaddingTopWithForeground();
271         final int parentBottom = bottom - top - getPaddingBottomWithForeground();
272 
273         for (int i = 0; i < count; i++) {
274             final View child = getChildAt(i);
275             if (child.getVisibility() != GONE) {
276                 final LayoutParams lp = (LayoutParams) child.getLayoutParams();
277 
278                 final int width = child.getMeasuredWidth();
279                 final int height = child.getMeasuredHeight();
280 
281                 int childLeft;
282                 int childTop;
283 
284                 int gravity = lp.gravity;
285                 if (gravity == -1) {
286                     gravity = DEFAULT_CHILD_GRAVITY;
287                 }
288 
289                 final int layoutDirection = getLayoutDirection();
290                 final int absoluteGravity = Gravity.getAbsoluteGravity(gravity, layoutDirection);
291                 final int verticalGravity = gravity & Gravity.VERTICAL_GRAVITY_MASK;
292 
293                 switch (absoluteGravity & Gravity.HORIZONTAL_GRAVITY_MASK) {
294                     case Gravity.CENTER_HORIZONTAL:
295                         childLeft = parentLeft + (parentRight - parentLeft - width) / 2 +
296                         lp.leftMargin - lp.rightMargin;
297                         break;
298                     case Gravity.RIGHT:
299                         if (!forceLeftGravity) {
300                             childLeft = parentRight - width - lp.rightMargin;
301                             break;
302                         }
303                     case Gravity.LEFT:
304                     default:
305                         childLeft = parentLeft + lp.leftMargin;
306                 }
307 
308                 switch (verticalGravity) {
309                     case Gravity.TOP:
310                         childTop = parentTop + lp.topMargin;
311                         break;
312                     case Gravity.CENTER_VERTICAL:
313                         childTop = parentTop + (parentBottom - parentTop - height) / 2 +
314                         lp.topMargin - lp.bottomMargin;
315                         break;
316                     case Gravity.BOTTOM:
317                         childTop = parentBottom - height - lp.bottomMargin;
318                         break;
319                     default:
320                         childTop = parentTop + lp.topMargin;
321                 }
322 
323                 child.layout(childLeft, childTop, childLeft + width, childTop + height);
324             }
325         }
326     }
327 
328     /**
329      * Sets whether to consider all children, or just those in
330      * the VISIBLE or INVISIBLE state, when measuring. Defaults to false.
331      *
332      * @param measureAll true to consider children marked GONE, false otherwise.
333      * Default value is false.
334      *
335      * @attr ref android.R.styleable#FrameLayout_measureAllChildren
336      */
337     @android.view.RemotableViewMethod
setMeasureAllChildren(boolean measureAll)338     public void setMeasureAllChildren(boolean measureAll) {
339         mMeasureAllChildren = measureAll;
340     }
341 
342     /**
343      * Determines whether all children, or just those in the VISIBLE or
344      * INVISIBLE state, are considered when measuring.
345      *
346      * @return Whether all children are considered when measuring.
347      *
348      * @deprecated This method is deprecated in favor of
349      * {@link #getMeasureAllChildren() getMeasureAllChildren()}, which was
350      * renamed for consistency with
351      * {@link #setMeasureAllChildren(boolean) setMeasureAllChildren()}.
352      */
353     @Deprecated
getConsiderGoneChildrenWhenMeasuring()354     public boolean getConsiderGoneChildrenWhenMeasuring() {
355         return getMeasureAllChildren();
356     }
357 
358     /**
359      * Determines whether all children, or just those in the VISIBLE or
360      * INVISIBLE state, are considered when measuring.
361      *
362      * @return Whether all children are considered when measuring.
363      */
getMeasureAllChildren()364     public boolean getMeasureAllChildren() {
365         return mMeasureAllChildren;
366     }
367 
368     @Override
generateLayoutParams(AttributeSet attrs)369     public LayoutParams generateLayoutParams(AttributeSet attrs) {
370         return new FrameLayout.LayoutParams(getContext(), attrs);
371     }
372 
373     @Override
shouldDelayChildPressedState()374     public boolean shouldDelayChildPressedState() {
375         return false;
376     }
377 
378     @Override
checkLayoutParams(ViewGroup.LayoutParams p)379     protected boolean checkLayoutParams(ViewGroup.LayoutParams p) {
380         return p instanceof LayoutParams;
381     }
382 
383     @Override
generateLayoutParams(ViewGroup.LayoutParams lp)384     protected ViewGroup.LayoutParams generateLayoutParams(ViewGroup.LayoutParams lp) {
385         if (sPreserveMarginParamsInLayoutParamConversion) {
386             if (lp instanceof LayoutParams) {
387                 return new LayoutParams((LayoutParams) lp);
388             } else if (lp instanceof MarginLayoutParams) {
389                 return new LayoutParams((MarginLayoutParams) lp);
390             }
391         }
392         return new LayoutParams(lp);
393     }
394 
395     @Override
getAccessibilityClassName()396     public CharSequence getAccessibilityClassName() {
397         return FrameLayout.class.getName();
398     }
399 
400     /** @hide */
401     @Override
encodeProperties(@onNull ViewHierarchyEncoder encoder)402     protected void encodeProperties(@NonNull ViewHierarchyEncoder encoder) {
403         super.encodeProperties(encoder);
404 
405         encoder.addProperty("measurement:measureAllChildren", mMeasureAllChildren);
406         encoder.addProperty("padding:foregroundPaddingLeft", mForegroundPaddingLeft);
407         encoder.addProperty("padding:foregroundPaddingTop", mForegroundPaddingTop);
408         encoder.addProperty("padding:foregroundPaddingRight", mForegroundPaddingRight);
409         encoder.addProperty("padding:foregroundPaddingBottom", mForegroundPaddingBottom);
410     }
411 
412     /**
413      * Per-child layout information for layouts that support margins.
414      * See {@link android.R.styleable#FrameLayout_Layout FrameLayout Layout Attributes}
415      * for a list of all child view attributes that this class supports.
416      *
417      * @attr ref android.R.styleable#FrameLayout_Layout_layout_gravity
418      */
419     public static class LayoutParams extends MarginLayoutParams {
420         /**
421          * Value for {@link #gravity} indicating that a gravity has not been
422          * explicitly specified.
423          */
424         public static final int UNSPECIFIED_GRAVITY = -1;
425 
426         /**
427          * The gravity to apply with the View to which these layout parameters
428          * are associated.
429          * <p>
430          * The default value is {@link #UNSPECIFIED_GRAVITY}, which is treated
431          * by FrameLayout as {@code Gravity.TOP | Gravity.START}.
432          *
433          * @see android.view.Gravity
434          * @attr ref android.R.styleable#FrameLayout_Layout_layout_gravity
435          */
436         public int gravity = UNSPECIFIED_GRAVITY;
437 
LayoutParams(@onNull Context c, @Nullable AttributeSet attrs)438         public LayoutParams(@NonNull Context c, @Nullable AttributeSet attrs) {
439             super(c, attrs);
440 
441             final TypedArray a = c.obtainStyledAttributes(attrs, R.styleable.FrameLayout_Layout);
442             gravity = a.getInt(R.styleable.FrameLayout_Layout_layout_gravity, UNSPECIFIED_GRAVITY);
443             a.recycle();
444         }
445 
LayoutParams(int width, int height)446         public LayoutParams(int width, int height) {
447             super(width, height);
448         }
449 
450         /**
451          * Creates a new set of layout parameters with the specified width, height
452          * and weight.
453          *
454          * @param width the width, either {@link #MATCH_PARENT},
455          *              {@link #WRAP_CONTENT} or a fixed size in pixels
456          * @param height the height, either {@link #MATCH_PARENT},
457          *               {@link #WRAP_CONTENT} or a fixed size in pixels
458          * @param gravity the gravity
459          *
460          * @see android.view.Gravity
461          */
LayoutParams(int width, int height, int gravity)462         public LayoutParams(int width, int height, int gravity) {
463             super(width, height);
464             this.gravity = gravity;
465         }
466 
LayoutParams(@onNull ViewGroup.LayoutParams source)467         public LayoutParams(@NonNull ViewGroup.LayoutParams source) {
468             super(source);
469         }
470 
LayoutParams(@onNull ViewGroup.MarginLayoutParams source)471         public LayoutParams(@NonNull ViewGroup.MarginLayoutParams source) {
472             super(source);
473         }
474 
475         /**
476          * Copy constructor. Clones the width, height, margin values, and
477          * gravity of the source.
478          *
479          * @param source The layout params to copy from.
480          */
LayoutParams(@onNull LayoutParams source)481         public LayoutParams(@NonNull LayoutParams source) {
482             super(source);
483 
484             this.gravity = source.gravity;
485         }
486     }
487 }
488