19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/*
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2006 The Android Open Source Project
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License.
69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at
79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project *
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and
149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License.
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpackage android.widget;
189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
196876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guyimport android.util.ArrayMap;
209e3b002d3f9141d54948a65e0330fdcd09e75a30Fabrice Di Meglioimport com.android.internal.R;
219e3b002d3f9141d54948a65e0330fdcd09e75a30Fabrice Di Meglio
22bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guyimport java.util.ArrayDeque;
239e3b002d3f9141d54948a65e0330fdcd09e75a30Fabrice Di Meglioimport java.util.ArrayList;
249e3b002d3f9141d54948a65e0330fdcd09e75a30Fabrice Di Meglioimport java.util.Comparator;
259e3b002d3f9141d54948a65e0330fdcd09e75a30Fabrice Di Meglioimport java.util.SortedSet;
269e3b002d3f9141d54948a65e0330fdcd09e75a30Fabrice Di Meglioimport java.util.TreeSet;
279e3b002d3f9141d54948a65e0330fdcd09e75a30Fabrice Di Meglio
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.Context;
296a03640539405afbdefe72894759281b98aa6e6fFabrice Di Meglioimport android.content.res.TypedArray;
3075986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganovimport android.graphics.Rect;
317da4b73a236b7c72d1337696949df7a00776dd06Adam Powellimport android.os.Build;
329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.util.AttributeSet;
336876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guyimport android.util.Pools.SynchronizedPool;
346a03640539405afbdefe72894759281b98aa6e6fFabrice Di Meglioimport android.util.SparseArray;
359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.view.Gravity;
3675986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganovimport android.view.View;
37c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Projectimport android.view.ViewDebug;
3875986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganovimport android.view.ViewGroup;
3975986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganovimport android.view.accessibility.AccessibilityEvent;
408a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganovimport android.view.accessibility.AccessibilityNodeInfo;
419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.widget.RemoteViews.RemoteView;
426a03640539405afbdefe72894759281b98aa6e6fFabrice Di Meglio
43c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglioimport static android.os.Build.VERSION_CODES.JELLY_BEAN_MR1;
449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/**
469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * A Layout where the positions of the children can be described in relation to each other or to the
47520c4204fb9d4f2a7a79dcb5d171e681b7bc6a07Romain Guy * parent.
48a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy *
499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <p>
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Note that you cannot have a circular dependency between the size of the RelativeLayout and the
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * position of its children. For example, you cannot have a RelativeLayout whose height is set to
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link android.view.ViewGroup.LayoutParams#WRAP_CONTENT WRAP_CONTENT} and a child set to
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link #ALIGN_PARENT_BOTTOM}.
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * </p>
55a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy *
562c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * <p><strong>Note:</strong> In platform version 17 and lower, RelativeLayout was affected by
572c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * a measurement bug that could cause child views to be measured with incorrect
582c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * {@link android.view.View.MeasureSpec MeasureSpec} values. (See
592c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * {@link android.view.View.MeasureSpec#makeMeasureSpec(int, int) MeasureSpec.makeMeasureSpec}
602c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * for more details.) This was triggered when a RelativeLayout container was placed in
612c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * a scrolling container, such as a ScrollView or HorizontalScrollView. If a custom view
622c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * not equipped to properly measure with the MeasureSpec mode
632c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * {@link android.view.View.MeasureSpec#UNSPECIFIED UNSPECIFIED} was placed in a RelativeLayout,
642c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * this would silently work anyway as RelativeLayout would pass a very large
652c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * {@link android.view.View.MeasureSpec#AT_MOST AT_MOST} MeasureSpec instead.</p>
662c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell *
672c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * <p>This behavior has been preserved for apps that set <code>android:targetSdkVersion="17"</code>
682c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * or older in their manifest's <code>uses-sdk</code> tag for compatibility. Apps targeting SDK
692c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell * version 18 or newer will receive the correct behavior</p>
702c8cc972c3b03b76a80db517dde1e68895bbdd66Adam Powell *
714c359b76f9a030f92a302ba74a528faa170bad4eScott Main * <p>See the <a href="{@docRoot}guide/topics/ui/layout/relative.html">Relative
724c359b76f9a030f92a302ba74a528faa170bad4eScott Main * Layout</a> guide.</p>
7341ec65355bd6ded652769725b276d47c54a0d913Scott Main *
749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <p>
759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Also see {@link android.widget.RelativeLayout.LayoutParams RelativeLayout.LayoutParams} for
769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * layout attributes
779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * </p>
78a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy *
799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#RelativeLayout_gravity
809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @attr ref android.R.styleable#RelativeLayout_ignoreGravity
819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */
829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project@RemoteView
839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpublic class RelativeLayout extends ViewGroup {
849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int TRUE = -1;
859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns a child's right edge with another child's left edge.
889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int LEFT_OF                  = 0;
909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns a child's left edge with another child's right edge.
929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int RIGHT_OF                 = 1;
949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns a child's bottom edge with another child's top edge.
969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int ABOVE                    = 2;
989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns a child's top edge with another child's bottom edge.
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int BELOW                    = 3;
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns a child's baseline with another child's baseline.
1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int ALIGN_BASELINE           = 4;
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns a child's left edge with another child's left edge.
1099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int ALIGN_LEFT               = 5;
1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns a child's top edge with another child's top edge.
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int ALIGN_TOP                = 6;
1159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns a child's right edge with another child's right edge.
1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int ALIGN_RIGHT              = 7;
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns a child's bottom edge with another child's bottom edge.
1219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int ALIGN_BOTTOM             = 8;
1239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns the child's left edge with its RelativeLayout
1269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * parent's left edge.
1279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int ALIGN_PARENT_LEFT        = 9;
1299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns the child's top edge with its RelativeLayout
1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * parent's top edge.
1329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int ALIGN_PARENT_TOP         = 10;
1349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns the child's right edge with its RelativeLayout
1369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * parent's right edge.
1379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int ALIGN_PARENT_RIGHT       = 11;
1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that aligns the child's bottom edge with its RelativeLayout
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * parent's bottom edge.
1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int ALIGN_PARENT_BOTTOM      = 12;
1449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that centers the child with respect to the bounds of its
1479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * RelativeLayout parent.
1489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int CENTER_IN_PARENT         = 13;
1509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that centers the child horizontally with respect to the
1529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * bounds of its RelativeLayout parent.
1539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int CENTER_HORIZONTAL        = 14;
1559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
1569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Rule that centers the child vertically with respect to the
1579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * bounds of its RelativeLayout parent.
1589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
1599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static final int CENTER_VERTICAL          = 15;
160f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    /**
161f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * Rule that aligns a child's end edge with another child's start edge.
162f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     */
163f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    public static final int START_OF                 = 16;
164f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    /**
165f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * Rule that aligns a child's start edge with another child's end edge.
166f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     */
167f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    public static final int END_OF                   = 17;
168f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    /**
169f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * Rule that aligns a child's start edge with another child's start edge.
170f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     */
171f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    public static final int ALIGN_START              = 18;
172f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    /**
173f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * Rule that aligns a child's end edge with another child's end edge.
174f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     */
175f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    public static final int ALIGN_END                = 19;
176f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    /**
177f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * Rule that aligns the child's start edge with its RelativeLayout
178f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * parent's start edge.
179f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     */
180f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    public static final int ALIGN_PARENT_START       = 20;
181f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    /**
182f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * Rule that aligns the child's end edge with its RelativeLayout
183f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * parent's end edge.
184f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     */
185f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    public static final int ALIGN_PARENT_END         = 21;
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
187f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    private static final int VERB_COUNT              = 22;
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
189bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy
190bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy    private static final int[] RULES_VERTICAL = {
191bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy            ABOVE, BELOW, ALIGN_BASELINE, ALIGN_TOP, ALIGN_BOTTOM
192bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy    };
193bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy
194bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy    private static final int[] RULES_HORIZONTAL = {
195f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            LEFT_OF, RIGHT_OF, ALIGN_LEFT, ALIGN_RIGHT, START_OF, END_OF, ALIGN_START, ALIGN_END
196bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy    };
197bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy
1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private View mBaselineView = null;
1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private boolean mHasBaselineAlignedChild;
2009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
201f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio    private int mGravity = Gravity.START | Gravity.TOP;
2029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private final Rect mContentBounds = new Rect();
2039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private final Rect mSelfBounds = new Rect();
2049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int mIgnoreGravity;
2059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
206725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    private SortedSet<View> mTopToBottomLeftToRightSet = null;
207725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
208725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    private boolean mDirtyHierarchy;
2096876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy    private View[] mSortedHorizontalChildren;
2106876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy    private View[] mSortedVerticalChildren;
211725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    private final DependencyGraph mGraph = new DependencyGraph();
21275986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov
2137da4b73a236b7c72d1337696949df7a00776dd06Adam Powell    // Compatibility hack. Old versions of the platform had problems
2147da4b73a236b7c72d1337696949df7a00776dd06Adam Powell    // with MeasureSpec value overflow and RelativeLayout was one source of them.
2157da4b73a236b7c72d1337696949df7a00776dd06Adam Powell    // Some apps came to rely on them. :(
2167da4b73a236b7c72d1337696949df7a00776dd06Adam Powell    private boolean mAllowBrokenMeasureSpecs = false;
21776d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy    // Compatibility hack. Old versions of the platform would not take
21876d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy    // margins and padding into account when generating the height measure spec
21976d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy    // for children during the horizontal measure pass.
22076d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy    private boolean mMeasureVerticalWithPaddingMargin = false;
2217da4b73a236b7c72d1337696949df7a00776dd06Adam Powell
222d5ffc79fc8c94df58509c4881b18e64cd922960eFabrice Di Meglio    // A default width used for RTL measure pass
223ca2e9e1122ba8b83d05bd144d0be31e8a3c30537Philip Milne    /**
224ca2e9e1122ba8b83d05bd144d0be31e8a3c30537Philip Milne     * Value reduced so as not to interfere with View's measurement spec. flags. See:
225ca2e9e1122ba8b83d05bd144d0be31e8a3c30537Philip Milne     * {@link View#MEASURED_SIZE_MASK}.
226ca2e9e1122ba8b83d05bd144d0be31e8a3c30537Philip Milne     * {@link View#MEASURED_STATE_TOO_SMALL}.
227ca2e9e1122ba8b83d05bd144d0be31e8a3c30537Philip Milne     **/
2283c647d23e5e00b5663f344b7c3f0ab38d74cf720Philip Milne    private static final int DEFAULT_WIDTH = 0x00010000;
229306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio
2309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public RelativeLayout(Context context) {
2319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super(context);
23276d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy        queryCompatibilityModes(context);
2339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public RelativeLayout(Context context, AttributeSet attrs) {
2369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super(context, attrs);
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        initFromAttributes(context, attrs);
23876d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy        queryCompatibilityModes(context);
2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public RelativeLayout(Context context, AttributeSet attrs, int defStyle) {
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        super(context, attrs, defStyle);
2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        initFromAttributes(context, attrs);
24476d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy        queryCompatibilityModes(context);
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void initFromAttributes(Context context, AttributeSet attrs) {
2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.RelativeLayout);
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mIgnoreGravity = a.getResourceId(R.styleable.RelativeLayout_ignoreGravity, View.NO_ID);
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mGravity = a.getInt(R.styleable.RelativeLayout_gravity, mGravity);
2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        a.recycle();
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
25476d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy    private void queryCompatibilityModes(Context context) {
25576d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy        int version = context.getApplicationInfo().targetSdkVersion;
25676d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy        mAllowBrokenMeasureSpecs = version <= Build.VERSION_CODES.JELLY_BEAN_MR1;
25776d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy        mMeasureVerticalWithPaddingMargin = version >= Build.VERSION_CODES.JELLY_BEAN_MR2;
25876d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy    }
25976d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy
260e0a799a2ac1ca78e30fbac9e4e12a063425c08d3Patrick Dubroy    @Override
261e0a799a2ac1ca78e30fbac9e4e12a063425c08d3Patrick Dubroy    public boolean shouldDelayChildPressedState() {
262e0a799a2ac1ca78e30fbac9e4e12a063425c08d3Patrick Dubroy        return false;
263e0a799a2ac1ca78e30fbac9e4e12a063425c08d3Patrick Dubroy    }
264e0a799a2ac1ca78e30fbac9e4e12a063425c08d3Patrick Dubroy
2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Defines which View is ignored when the gravity is applied. This setting has no
267f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * effect if the gravity is <code>Gravity.START | Gravity.TOP</code>.
2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param viewId The id of the View to be ignored by gravity, or 0 if no View
2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *        should be ignored.
2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #setGravity(int)
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_ignoreGravity
2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setIgnoreGravity(int viewId) {
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mIgnoreGravity = viewId;
2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
2821018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne     * Describes how the child views are positioned.
2831018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne     *
2841018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne     * @return the gravity.
2851018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne     *
2861018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne     * @see #setGravity(int)
2871018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne     * @see android.view.Gravity
2881018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne     *
2891018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne     * @attr ref android.R.styleable#RelativeLayout_gravity
2901018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne     */
2911018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne    public int getGravity() {
2921018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne        return mGravity;
2931018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne    }
2941018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne
2951018fb42cb4958511a141787705aa429c5ec9bd3Philip Milne    /**
2969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Describes how the child views are positioned. Defaults to
297f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * <code>Gravity.START | Gravity.TOP</code>.
2989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
2991fec24e0fff95afaa90bbf811e9f8db659eb9324Adam Powell     * <p>Note that since RelativeLayout considers the positioning of each child
3001fec24e0fff95afaa90bbf811e9f8db659eb9324Adam Powell     * relative to one another to be significant, setting gravity will affect
3011fec24e0fff95afaa90bbf811e9f8db659eb9324Adam Powell     * the positioning of all children as a single unit within the parent.
3021fec24e0fff95afaa90bbf811e9f8db659eb9324Adam Powell     * This happens after children have been relatively positioned.</p>
3031fec24e0fff95afaa90bbf811e9f8db659eb9324Adam Powell     *
3049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param gravity See {@link android.view.Gravity}
3059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #setHorizontalGravity(int)
3079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @see #setVerticalGravity(int)
3089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
3099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_gravity
3109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
3119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
3129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setGravity(int gravity) {
3139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mGravity != gravity) {
3146a03640539405afbdefe72894759281b98aa6e6fFabrice Di Meglio            if ((gravity & Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK) == 0) {
3159e3b002d3f9141d54948a65e0330fdcd09e75a30Fabrice Di Meglio                gravity |= Gravity.START;
3169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
3179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if ((gravity & Gravity.VERTICAL_GRAVITY_MASK) == 0) {
3199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                gravity |= Gravity.TOP;
3209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mGravity = gravity;
3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            requestLayout();
3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setHorizontalGravity(int horizontalGravity) {
3296a03640539405afbdefe72894759281b98aa6e6fFabrice Di Meglio        final int gravity = horizontalGravity & Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK;
3306a03640539405afbdefe72894759281b98aa6e6fFabrice Di Meglio        if ((mGravity & Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK) != gravity) {
3316a03640539405afbdefe72894759281b98aa6e6fFabrice Di Meglio            mGravity = (mGravity & ~Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK) | gravity;
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            requestLayout();
3339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @android.view.RemotableViewMethod
3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public void setVerticalGravity(int verticalGravity) {
3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final int gravity = verticalGravity & Gravity.VERTICAL_GRAVITY_MASK;
3399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if ((mGravity & Gravity.VERTICAL_GRAVITY_MASK) != gravity) {
3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mGravity = (mGravity & ~Gravity.VERTICAL_GRAVITY_MASK) | gravity;
3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            requestLayout();
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public int getBaseline() {
3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return mBaselineView != null ? mBaselineView.getBaseline() : super.getBaseline();
3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
351725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    public void requestLayout() {
352725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        super.requestLayout();
353725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        mDirtyHierarchy = true;
354725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    }
355725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
356725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    private void sortChildren() {
3576876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy        final int count = getChildCount();
3586876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy        if (mSortedVerticalChildren == null || mSortedVerticalChildren.length != count) {
3596876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy            mSortedVerticalChildren = new View[count];
3606876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy        }
3616876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy
3626876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy        if (mSortedHorizontalChildren == null || mSortedHorizontalChildren.length != count) {
3636876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy            mSortedHorizontalChildren = new View[count];
3646876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy        }
365725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
366725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        final DependencyGraph graph = mGraph;
367725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        graph.clear();
368725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
369725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        for (int i = 0; i < count; i++) {
3706876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy            graph.add(getChildAt(i));
371725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        }
372725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
373bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy        graph.getSortedViews(mSortedVerticalChildren, RULES_VERTICAL);
374bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy        graph.getSortedViews(mSortedHorizontalChildren, RULES_HORIZONTAL);
375725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    }
376725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
377725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    @Override
3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
379725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        if (mDirtyHierarchy) {
380725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            mDirtyHierarchy = false;
381725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            sortChildren();
382725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        }
383725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int myWidth = -1;
3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int myHeight = -1;
3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int width = 0;
3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int height = 0;
3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
390132a742b94b9716451ddef30cec20548b346f1b9Adam Powell        final int widthMode = MeasureSpec.getMode(widthMeasureSpec);
391132a742b94b9716451ddef30cec20548b346f1b9Adam Powell        final int heightMode = MeasureSpec.getMode(heightMeasureSpec);
392132a742b94b9716451ddef30cec20548b346f1b9Adam Powell        final int widthSize = MeasureSpec.getSize(widthMeasureSpec);
393132a742b94b9716451ddef30cec20548b346f1b9Adam Powell        final int heightSize = MeasureSpec.getSize(heightMeasureSpec);
3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Record our dimensions if they are known;
3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (widthMode != MeasureSpec.UNSPECIFIED) {
3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            myWidth = widthSize;
3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (heightMode != MeasureSpec.UNSPECIFIED) {
4019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            myHeight = heightSize;
4029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (widthMode == MeasureSpec.EXACTLY) {
4059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            width = myWidth;
4069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (heightMode == MeasureSpec.EXACTLY) {
4099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            height = myHeight;
4109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        mHasBaselineAlignedChild = false;
4139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        View ignore = null;
4156a03640539405afbdefe72894759281b98aa6e6fFabrice Di Meglio        int gravity = mGravity & Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK;
416f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        final boolean horizontalGravity = gravity != Gravity.START && gravity != 0;
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        gravity = mGravity & Gravity.VERTICAL_GRAVITY_MASK;
4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        final boolean verticalGravity = gravity != Gravity.TOP && gravity != 0;
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int left = Integer.MAX_VALUE;
4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int top = Integer.MAX_VALUE;
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int right = Integer.MIN_VALUE;
4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int bottom = Integer.MIN_VALUE;
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
425f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy        boolean offsetHorizontalAxis = false;
426f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy        boolean offsetVerticalAxis = false;
427f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy
4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if ((horizontalGravity || verticalGravity) && mIgnoreGravity != View.NO_ID) {
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            ignore = findViewById(mIgnoreGravity);
4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
432f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy        final boolean isWrapContentWidth = widthMode != MeasureSpec.EXACTLY;
433f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy        final boolean isWrapContentHeight = heightMode != MeasureSpec.EXACTLY;
434f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy
435306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio        // We need to know our size for doing the correct computation of children positioning in RTL
436306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio        // mode but there is no practical way to get it instead of running the code below.
437d5ffc79fc8c94df58509c4881b18e64cd922960eFabrice Di Meglio        // So, instead of running the code twice, we just set the width to a "default display width"
438306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio        // before the computation and then, as a last pass, we will update their real position with
439d5ffc79fc8c94df58509c4881b18e64cd922960eFabrice Di Meglio        // an offset equals to "DEFAULT_WIDTH - width".
440306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio        final int layoutDirection = getLayoutDirection();
441306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio        if (isLayoutRtl() && myWidth == -1) {
442d5ffc79fc8c94df58509c4881b18e64cd922960eFabrice Di Meglio            myWidth = DEFAULT_WIDTH;
443306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio        }
444306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio
445e24ef6032f52cb754bfeb9ab32aae0a5cfa61f8aRomain Guy        View[] views = mSortedHorizontalChildren;
446725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        int count = views.length;
44713705eda8cc065ee98feb94af649615579cd4a58Fabrice Di Meglio
448725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        for (int i = 0; i < count; i++) {
449725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            View child = views[i];
450725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            if (child.getVisibility() != GONE) {
451725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                LayoutParams params = (LayoutParams) child.getLayoutParams();
452547261320cb4bbb3e967246bc86578263d94e84dFabrice Di Meglio                int[] rules = params.getRules(layoutDirection);
453956070383945db5f842ec05e507fd0233705738cRomain Guy
454547261320cb4bbb3e967246bc86578263d94e84dFabrice Di Meglio                applyHorizontalSizeRules(params, myWidth, rules);
455f782e60efc09f210643432f31b4c18026d7716d6Romain Guy                measureChildHorizontal(child, params, myWidth, myHeight);
456547261320cb4bbb3e967246bc86578263d94e84dFabrice Di Meglio
457f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                if (positionChildHorizontal(child, params, myWidth, isWrapContentWidth)) {
458f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    offsetHorizontalAxis = true;
459f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                }
460725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            }
461725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        }
462725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
463e24ef6032f52cb754bfeb9ab32aae0a5cfa61f8aRomain Guy        views = mSortedVerticalChildren;
464725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        count = views.length;
4652bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio        final int targetSdkVersion = getContext().getApplicationInfo().targetSdkVersion;
466f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy
467725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        for (int i = 0; i < count; i++) {
468725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            View child = views[i];
4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (child.getVisibility() != GONE) {
4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                LayoutParams params = (LayoutParams) child.getLayoutParams();
471956070383945db5f842ec05e507fd0233705738cRomain Guy
472956070383945db5f842ec05e507fd0233705738cRomain Guy                applyVerticalSizeRules(params, myHeight);
4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                measureChild(child, params, myWidth, myHeight);
474f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                if (positionChildVertical(child, params, myHeight, isWrapContentHeight)) {
475f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    offsetVerticalAxis = true;
476f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                }
4779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
478f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                if (isWrapContentWidth) {
479306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio                    if (isLayoutRtl()) {
480e8222dddaf2e3da14380101e818d4254899e0c0dChet Haase                        if (targetSdkVersion < Build.VERSION_CODES.KITKAT) {
4812bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                            width = Math.max(width, myWidth - params.mLeft);
4822bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                        } else {
4832bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                            width = Math.max(width, myWidth - params.mLeft - params.leftMargin);
4842bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                        }
485306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio                    } else {
486e8222dddaf2e3da14380101e818d4254899e0c0dChet Haase                        if (targetSdkVersion < Build.VERSION_CODES.KITKAT) {
4872bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                            width = Math.max(width, params.mRight);
4882bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                        } else {
4892bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                            width = Math.max(width, params.mRight + params.rightMargin);
4902bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                        }
491306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio                    }
4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
493f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy
494f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                if (isWrapContentHeight) {
495e8222dddaf2e3da14380101e818d4254899e0c0dChet Haase                    if (targetSdkVersion < Build.VERSION_CODES.KITKAT) {
4962bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                        height = Math.max(height, params.mBottom);
4972bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                    } else {
4982bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                        height = Math.max(height, params.mBottom + params.bottomMargin);
4992bd961ae38ebd0acb7d33b32144a3c8a0949c023Fabrice Di Meglio                    }
5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (child != ignore || verticalGravity) {
5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    left = Math.min(left, params.mLeft - params.leftMargin);
5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    top = Math.min(top, params.mTop - params.topMargin);
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (child != ignore || horizontalGravity) {
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    right = Math.max(right, params.mRight + params.rightMargin);
5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    bottom = Math.max(bottom, params.mBottom + params.bottomMargin);
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mHasBaselineAlignedChild) {
515725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            for (int i = 0; i < count; i++) {
5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                View child = getChildAt(i);
5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (child.getVisibility() != GONE) {
5189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    LayoutParams params = (LayoutParams) child.getLayoutParams();
5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    alignBaseline(child, params);
5209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (child != ignore || verticalGravity) {
522725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                        left = Math.min(left, params.mLeft - params.leftMargin);
523725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                        top = Math.min(top, params.mTop - params.topMargin);
5249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
5259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (child != ignore || horizontalGravity) {
5279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        right = Math.max(right, params.mRight + params.rightMargin);
5289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        bottom = Math.max(bottom, params.mBottom + params.bottomMargin);
5299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
5309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
5319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
534f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy        if (isWrapContentWidth) {
535a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy            // Width already has left padding in it since it was calculated by looking at
5369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // the right of each child view
5379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            width += mPaddingRight;
5389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
539758d5a4563a71d4f3f7efbf7e450e8e537abc185Adam Powell            if (mLayoutParams != null && mLayoutParams.width >= 0) {
5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                width = Math.max(width, mLayoutParams.width);
5419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            width = Math.max(width, getSuggestedMinimumWidth());
5449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            width = resolveSize(width, widthMeasureSpec);
545f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy
546f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy            if (offsetHorizontalAxis) {
547d10a576791675628a014c0488c3d054371d4d63aRomain Guy                for (int i = 0; i < count; i++) {
548f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    View child = getChildAt(i);
549f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    if (child.getVisibility() != GONE) {
550f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                        LayoutParams params = (LayoutParams) child.getLayoutParams();
551f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        final int[] rules = params.getRules(layoutDirection);
552f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                        if (rules[CENTER_IN_PARENT] != 0 || rules[CENTER_HORIZONTAL] != 0) {
553f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                            centerHorizontal(child, params, width);
55442460ac1bb5512a17a6891f7d99e2b45db0889d8Romain Guy                        } else if (rules[ALIGN_PARENT_RIGHT] != 0) {
55542460ac1bb5512a17a6891f7d99e2b45db0889d8Romain Guy                            final int childWidth = child.getMeasuredWidth();
55642460ac1bb5512a17a6891f7d99e2b45db0889d8Romain Guy                            params.mLeft = width - mPaddingRight - childWidth;
55742460ac1bb5512a17a6891f7d99e2b45db0889d8Romain Guy                            params.mRight = params.mLeft + childWidth;
558f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                        }
559f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    }
560f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                }
561f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy            }
5629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
563f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy
564f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy        if (isWrapContentHeight) {
565a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy            // Height already has top padding in it since it was calculated by looking at
5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // the bottom of each child view
5679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            height += mPaddingBottom;
5689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
569758d5a4563a71d4f3f7efbf7e450e8e537abc185Adam Powell            if (mLayoutParams != null && mLayoutParams.height >= 0) {
5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                height = Math.max(height, mLayoutParams.height);
5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            height = Math.max(height, getSuggestedMinimumHeight());
5749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            height = resolveSize(height, heightMeasureSpec);
575f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy
576f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy            if (offsetVerticalAxis) {
577f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                for (int i = 0; i < count; i++) {
578f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    View child = getChildAt(i);
579f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    if (child.getVisibility() != GONE) {
580f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                        LayoutParams params = (LayoutParams) child.getLayoutParams();
581f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        final int[] rules = params.getRules(layoutDirection);
582f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                        if (rules[CENTER_IN_PARENT] != 0 || rules[CENTER_VERTICAL] != 0) {
583f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                            centerVertical(child, params, height);
58442460ac1bb5512a17a6891f7d99e2b45db0889d8Romain Guy                        } else if (rules[ALIGN_PARENT_BOTTOM] != 0) {
58542460ac1bb5512a17a6891f7d99e2b45db0889d8Romain Guy                            final int childHeight = child.getMeasuredHeight();
58642460ac1bb5512a17a6891f7d99e2b45db0889d8Romain Guy                            params.mTop = height - mPaddingBottom - childHeight;
58742460ac1bb5512a17a6891f7d99e2b45db0889d8Romain Guy                            params.mBottom = params.mTop + childHeight;
588f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                        }
589f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    }
590f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                }
591f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy            }
5929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (horizontalGravity || verticalGravity) {
5959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            final Rect selfBounds = mSelfBounds;
5969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            selfBounds.set(mPaddingLeft, mPaddingTop, width - mPaddingRight,
5979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    height - mPaddingBottom);
5989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            final Rect contentBounds = mContentBounds;
6006a03640539405afbdefe72894759281b98aa6e6fFabrice Di Meglio            Gravity.apply(mGravity, right - left, bottom - top, selfBounds, contentBounds,
601c0053223bedf33581b0830fb87be32c1f26e5372Fabrice Di Meglio                    layoutDirection);
6029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            final int horizontalOffset = contentBounds.left - left;
6049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            final int verticalOffset = contentBounds.top - top;
6059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (horizontalOffset != 0 || verticalOffset != 0) {
606725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                for (int i = 0; i < count; i++) {
6079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    View child = getChildAt(i);
6089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    if (child.getVisibility() != GONE && child != ignore) {
6099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        LayoutParams params = (LayoutParams) child.getLayoutParams();
610d10a576791675628a014c0488c3d054371d4d63aRomain Guy                        if (horizontalGravity) {
611d10a576791675628a014c0488c3d054371d4d63aRomain Guy                            params.mLeft += horizontalOffset;
612d10a576791675628a014c0488c3d054371d4d63aRomain Guy                            params.mRight += horizontalOffset;
613d10a576791675628a014c0488c3d054371d4d63aRomain Guy                        }
614d10a576791675628a014c0488c3d054371d4d63aRomain Guy                        if (verticalGravity) {
615d10a576791675628a014c0488c3d054371d4d63aRomain Guy                            params.mTop += verticalOffset;
616d10a576791675628a014c0488c3d054371d4d63aRomain Guy                            params.mBottom += verticalOffset;
617d10a576791675628a014c0488c3d054371d4d63aRomain Guy                        }
6189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    }
6199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
6209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
623306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio        if (isLayoutRtl()) {
624306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio            final int offsetWidth = myWidth - width;
625306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio            for (int i = 0; i < count; i++) {
626306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio                View child = getChildAt(i);
627306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio                if (child.getVisibility() != GONE) {
628306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio                    LayoutParams params = (LayoutParams) child.getLayoutParams();
629306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio                    params.mLeft -= offsetWidth;
630306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio                    params.mRight -= offsetWidth;
631306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio                }
632306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio            }
633306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio
634306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio        }
635306fe5cc0e85e8a5ef07ff64493982f99caeba3aFabrice Di Meglio
6369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        setMeasuredDimension(width, height);
6379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private void alignBaseline(View child, LayoutParams params) {
640e56ffdc7b31b0937628609cc3bbaa15879023569Fabrice Di Meglio        final int layoutDirection = getLayoutDirection();
641f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        int[] rules = params.getRules(layoutDirection);
6429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int anchorBaseline = getRelatedViewBaseline(rules, ALIGN_BASELINE);
6439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (anchorBaseline != -1) {
6459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            LayoutParams anchorParams = getRelatedViewParams(rules, ALIGN_BASELINE);
6469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (anchorParams != null) {
6479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int offset = anchorParams.mTop + anchorBaseline;
6489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int baseline = child.getBaseline();
6499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (baseline != -1) {
6509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    offset -= baseline;
6519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
6529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int height = params.mBottom - params.mTop;
6539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                params.mTop = offset;
6549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                params.mBottom = params.mTop + height;
6559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (mBaselineView == null) {
6599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mBaselineView = child;
6609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
6619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            LayoutParams lp = (LayoutParams) mBaselineView.getLayoutParams();
6629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (params.mTop < lp.mTop || (params.mTop == lp.mTop && params.mLeft < lp.mLeft)) {
6639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                mBaselineView = child;
6649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
6659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
6669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Measure a child. The child should have left, top, right and bottom information
6709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * stored in its LayoutParams. If any of these values is -1 it means that the view
6719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * can extend up to the corresponding edge.
6729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
6739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param child Child to measure
6749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param params LayoutParams associated with child
6759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param myWidth Width of the the RelativeLayout
6769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param myHeight Height of the RelativeLayout
6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
678725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    private void measureChild(View child, LayoutParams params, int myWidth, int myHeight) {
679e8fb03c05899c00f55ab1990e1b16038e82d1167Romain Guy        int childWidthMeasureSpec = getChildMeasureSpec(params.mLeft,
680e8fb03c05899c00f55ab1990e1b16038e82d1167Romain Guy                params.mRight, params.width,
681e8fb03c05899c00f55ab1990e1b16038e82d1167Romain Guy                params.leftMargin, params.rightMargin,
682e8fb03c05899c00f55ab1990e1b16038e82d1167Romain Guy                mPaddingLeft, mPaddingRight,
683e8fb03c05899c00f55ab1990e1b16038e82d1167Romain Guy                myWidth);
6841b7c791adf6d5e113cd07fa1a9cbef310f79145aRomain Guy        int childHeightMeasureSpec = getChildMeasureSpec(params.mTop,
6851b7c791adf6d5e113cd07fa1a9cbef310f79145aRomain Guy                params.mBottom, params.height,
6861b7c791adf6d5e113cd07fa1a9cbef310f79145aRomain Guy                params.topMargin, params.bottomMargin,
6871b7c791adf6d5e113cd07fa1a9cbef310f79145aRomain Guy                mPaddingTop, mPaddingBottom,
6881b7c791adf6d5e113cd07fa1a9cbef310f79145aRomain Guy                myHeight);
6899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
6909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
692f782e60efc09f210643432f31b4c18026d7716d6Romain Guy    private void measureChildHorizontal(View child, LayoutParams params, int myWidth, int myHeight) {
693956070383945db5f842ec05e507fd0233705738cRomain Guy        int childWidthMeasureSpec = getChildMeasureSpec(params.mLeft,
694956070383945db5f842ec05e507fd0233705738cRomain Guy                params.mRight, params.width,
695956070383945db5f842ec05e507fd0233705738cRomain Guy                params.leftMargin, params.rightMargin,
696956070383945db5f842ec05e507fd0233705738cRomain Guy                mPaddingLeft, mPaddingRight,
697956070383945db5f842ec05e507fd0233705738cRomain Guy                myWidth);
69876d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy        int maxHeight = myHeight;
69976d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy        if (mMeasureVerticalWithPaddingMargin) {
70076d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy            maxHeight = Math.max(0, myHeight - mPaddingTop - mPaddingBottom -
70176d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy                    params.topMargin - params.bottomMargin);
70276d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy        }
703f782e60efc09f210643432f31b4c18026d7716d6Romain Guy        int childHeightMeasureSpec;
7047da4b73a236b7c72d1337696949df7a00776dd06Adam Powell        if (myHeight < 0 && !mAllowBrokenMeasureSpecs) {
705f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy            if (params.height >= 0) {
706f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy                childHeightMeasureSpec = MeasureSpec.makeMeasureSpec(
707f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy                        params.height, MeasureSpec.EXACTLY);
708f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy            } else {
709f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy                // Negative values in a mySize/myWidth/myWidth value in RelativeLayout measurement
710f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy                // is code for, "we got an unspecified mode in the RelativeLayout's measurespec."
711f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy                // Carry it forward.
712f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy                childHeightMeasureSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
713f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy            }
714132a742b94b9716451ddef30cec20548b346f1b9Adam Powell        } else if (params.width == LayoutParams.MATCH_PARENT) {
71576d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy            childHeightMeasureSpec = MeasureSpec.makeMeasureSpec(maxHeight, MeasureSpec.EXACTLY);
716f782e60efc09f210643432f31b4c18026d7716d6Romain Guy        } else {
71776d59a3b7b3ceea1d0854e588532cfc91b44201dRomain Guy            childHeightMeasureSpec = MeasureSpec.makeMeasureSpec(maxHeight, MeasureSpec.AT_MOST);
718f782e60efc09f210643432f31b4c18026d7716d6Romain Guy        }
719725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
720725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    }
721725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
7229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
7239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Get a measure spec that accounts for all of the constraints on this view.
724f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * This includes size constraints imposed by the RelativeLayout as well as
7259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * the View's desired dimension.
7269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
7279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param childStart The left or top field of the child's layout params
7289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param childEnd The right or bottom field of the child's layout params
7299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param childSize The child's desired size (the width or height field of
7309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *        the child's layout params)
7319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param startMargin The left or top margin
7329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param endMargin The right or bottom margin
7339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param startPadding mPaddingLeft or mPaddingTop
7349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param endPadding mPaddingRight or mPaddingBottom
7359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @param mySize The width or height of this view (the RelativeLayout)
7369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @return MeasureSpec for the child
7379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
7389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int getChildMeasureSpec(int childStart, int childEnd,
7399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int childSize, int startMargin, int endMargin, int startPadding,
7409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            int endPadding, int mySize) {
7417da4b73a236b7c72d1337696949df7a00776dd06Adam Powell        if (mySize < 0 && !mAllowBrokenMeasureSpecs) {
742f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy            if (childSize >= 0) {
743f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy                return MeasureSpec.makeMeasureSpec(childSize, MeasureSpec.EXACTLY);
744f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy            }
745132a742b94b9716451ddef30cec20548b346f1b9Adam Powell            // Negative values in a mySize/myWidth/myWidth value in RelativeLayout measurement
746132a742b94b9716451ddef30cec20548b346f1b9Adam Powell            // is code for, "we got an unspecified mode in the RelativeLayout's measurespec."
747132a742b94b9716451ddef30cec20548b346f1b9Adam Powell            // Carry it forward.
748132a742b94b9716451ddef30cec20548b346f1b9Adam Powell            return MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
749132a742b94b9716451ddef30cec20548b346f1b9Adam Powell        }
750132a742b94b9716451ddef30cec20548b346f1b9Adam Powell
7519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int childSpecMode = 0;
7529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int childSpecSize = 0;
7539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Figure out start and end bounds.
7559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int tempStart = childStart;
7569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int tempEnd = childEnd;
7579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // If the view did not express a layout constraint for an edge, use
7599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // view's margins and our padding
7609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (tempStart < 0) {
7619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            tempStart = startPadding + startMargin;
7629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (tempEnd < 0) {
7649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            tempEnd = mySize - endPadding - endMargin;
7659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
7669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // Figure out maximum size available to this view
7689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int maxAvailable = tempEnd - tempStart;
7699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (childStart >= 0 && childEnd >= 0) {
7719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Constraints fixed both edges, so child must be an exact size
7729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childSpecMode = MeasureSpec.EXACTLY;
7739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childSpecSize = maxAvailable;
7749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else {
7759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (childSize >= 0) {
7769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Child wanted an exact size. Give as much as possible
7779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                childSpecMode = MeasureSpec.EXACTLY;
7789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
7799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (maxAvailable >= 0) {
7809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // We have a maxmum size in this dimension.
7819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    childSpecSize = Math.min(maxAvailable, childSize);
7829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } else {
7839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // We can grow in this dimension.
7849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    childSpecSize = childSize;
7859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
786980a938c1c9a6a5791a8240e5a1e6638ab28dc77Romain Guy            } else if (childSize == LayoutParams.MATCH_PARENT) {
787f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                // Child wanted to be as big as possible. Give all available
7889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // space
7899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                childSpecMode = MeasureSpec.EXACTLY;
7909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                childSpecSize = maxAvailable;
7919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else if (childSize == LayoutParams.WRAP_CONTENT) {
7929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // Child wants to wrap content. Use AT_MOST
7939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // to communicate available space if we know
7949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                // our max size
7959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                if (maxAvailable >= 0) {
796f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    // We have a maximum size in this dimension.
7979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    childSpecMode = MeasureSpec.AT_MOST;
7989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    childSpecSize = maxAvailable;
7999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                } else {
8009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // We can grow in this dimension. Child can be as big as it
8019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    // wants
8029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    childSpecMode = MeasureSpec.UNSPECIFIED;
8039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    childSpecSize = 0;
8049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
8059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return MeasureSpec.makeMeasureSpec(childSpecSize, childSpecMode);
8099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
811f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy    private boolean positionChildHorizontal(View child, LayoutParams params, int myWidth,
812f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy            boolean wrapContent) {
813f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy
814e56ffdc7b31b0937628609cc3bbaa15879023569Fabrice Di Meglio        final int layoutDirection = getLayoutDirection();
815f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        int[] rules = params.getRules(layoutDirection);
8169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (params.mLeft < 0 && params.mRight >= 0) {
8189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Right is fixed, but left varies
8199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            params.mLeft = params.mRight - child.getMeasuredWidth();
8209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (params.mLeft >= 0 && params.mRight < 0) {
8219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Left is fixed, but right varies
8229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            params.mRight = params.mLeft + child.getMeasuredWidth();
8239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (params.mLeft < 0 && params.mRight < 0) {
8249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Both left and right vary
825f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy            if (rules[CENTER_IN_PARENT] != 0 || rules[CENTER_HORIZONTAL] != 0) {
826f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                if (!wrapContent) {
827f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    centerHorizontal(child, params, myWidth);
828f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                } else {
829f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    params.mLeft = mPaddingLeft + params.leftMargin;
830f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    params.mRight = params.mLeft + child.getMeasuredWidth();
831f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                }
832f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                return true;
8339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else {
834f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                // This is the default case. For RTL we start from the right and for LTR we start
835f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                // from the left. This will give LEFT/TOP for LTR and RIGHT/TOP for RTL.
836f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                if (isLayoutRtl()) {
837f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    params.mRight = myWidth - mPaddingRight- params.rightMargin;
838f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    params.mLeft = params.mRight - child.getMeasuredWidth();
839f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                } else {
840f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    params.mLeft = mPaddingLeft + params.leftMargin;
841f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    params.mRight = params.mLeft + child.getMeasuredWidth();
842f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                }
8439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
845f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        return rules[ALIGN_PARENT_END] != 0;
846725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    }
847725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
848f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy    private boolean positionChildVertical(View child, LayoutParams params, int myHeight,
849f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy            boolean wrapContent) {
850f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy
851725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        int[] rules = params.getRules();
8529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (params.mTop < 0 && params.mBottom >= 0) {
8549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Bottom is fixed, but top varies
8559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            params.mTop = params.mBottom - child.getMeasuredHeight();
8569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (params.mTop >= 0 && params.mBottom < 0) {
8579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Top is fixed, but bottom varies
8589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            params.mBottom = params.mTop + child.getMeasuredHeight();
8599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (params.mTop < 0 && params.mBottom < 0) {
8609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Both top and bottom vary
861f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy            if (rules[CENTER_IN_PARENT] != 0 || rules[CENTER_VERTICAL] != 0) {
862f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                if (!wrapContent) {
863f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    centerVertical(child, params, myHeight);
864f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                } else {
865f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    params.mTop = mPaddingTop + params.topMargin;
866f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                    params.mBottom = params.mTop + child.getMeasuredHeight();
867f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                }
868f7dabb088a474f821d1b07af9a51d063b4782537Romain Guy                return true;
8699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            } else {
8709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                params.mTop = mPaddingTop + params.topMargin;
8719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                params.mBottom = params.mTop + child.getMeasuredHeight();
8729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
87442460ac1bb5512a17a6891f7d99e2b45db0889d8Romain Guy        return rules[ALIGN_PARENT_BOTTOM] != 0;
8759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
877547261320cb4bbb3e967246bc86578263d94e84dFabrice Di Meglio    private void applyHorizontalSizeRules(LayoutParams childParams, int myWidth, int[] rules) {
8789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        RelativeLayout.LayoutParams anchorParams;
8799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // -1 indicated a "soft requirement" in that direction. For example:
8819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // left=10, right=-1 means the view must start at 10, but can go as far as it wants to the right
8829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // left =-1, right=10 means the view must end at 10, but can go as far as it wants to the left
8839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        // left=10, right=20 means the left and right ends are both fixed
8849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        childParams.mLeft = -1;
8859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        childParams.mRight = -1;
8869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        anchorParams = getRelatedViewParams(rules, LEFT_OF);
8889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (anchorParams != null) {
8899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mRight = anchorParams.mLeft - (anchorParams.leftMargin +
8909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    childParams.rightMargin);
8919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (childParams.alignWithParent && rules[LEFT_OF] != 0) {
8929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (myWidth >= 0) {
8939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                childParams.mRight = myWidth - mPaddingRight - childParams.rightMargin;
8949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
8959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        anchorParams = getRelatedViewParams(rules, RIGHT_OF);
8989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (anchorParams != null) {
8999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mLeft = anchorParams.mRight + (anchorParams.rightMargin +
9009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    childParams.leftMargin);
9019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (childParams.alignWithParent && rules[RIGHT_OF] != 0) {
9029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mLeft = mPaddingLeft + childParams.leftMargin;
9039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        anchorParams = getRelatedViewParams(rules, ALIGN_LEFT);
9069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (anchorParams != null) {
9079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mLeft = anchorParams.mLeft + childParams.leftMargin;
9089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (childParams.alignWithParent && rules[ALIGN_LEFT] != 0) {
9099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mLeft = mPaddingLeft + childParams.leftMargin;
9109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        anchorParams = getRelatedViewParams(rules, ALIGN_RIGHT);
9139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (anchorParams != null) {
9149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mRight = anchorParams.mRight - childParams.rightMargin;
9159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (childParams.alignWithParent && rules[ALIGN_RIGHT] != 0) {
9169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (myWidth >= 0) {
9179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                childParams.mRight = myWidth - mPaddingRight - childParams.rightMargin;
9189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
9199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9211b7c791adf6d5e113cd07fa1a9cbef310f79145aRomain Guy        if (0 != rules[ALIGN_PARENT_LEFT]) {
9229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mLeft = mPaddingLeft + childParams.leftMargin;
9239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9251b7c791adf6d5e113cd07fa1a9cbef310f79145aRomain Guy        if (0 != rules[ALIGN_PARENT_RIGHT]) {
9269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (myWidth >= 0) {
9279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                childParams.mRight = myWidth - mPaddingRight - childParams.rightMargin;
9289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
9299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
930725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    }
931725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
932725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    private void applyVerticalSizeRules(LayoutParams childParams, int myHeight) {
933725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        int[] rules = childParams.getRules();
934725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        RelativeLayout.LayoutParams anchorParams;
9359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        childParams.mTop = -1;
9379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        childParams.mBottom = -1;
9389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        anchorParams = getRelatedViewParams(rules, ABOVE);
9409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (anchorParams != null) {
9419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mBottom = anchorParams.mTop - (anchorParams.topMargin +
9429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    childParams.bottomMargin);
9439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (childParams.alignWithParent && rules[ABOVE] != 0) {
9449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (myHeight >= 0) {
9459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                childParams.mBottom = myHeight - mPaddingBottom - childParams.bottomMargin;
9469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
9479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        anchorParams = getRelatedViewParams(rules, BELOW);
9509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (anchorParams != null) {
9519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mTop = anchorParams.mBottom + (anchorParams.bottomMargin +
9529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    childParams.topMargin);
9539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (childParams.alignWithParent && rules[BELOW] != 0) {
9549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mTop = mPaddingTop + childParams.topMargin;
9559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        anchorParams = getRelatedViewParams(rules, ALIGN_TOP);
9589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (anchorParams != null) {
9599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mTop = anchorParams.mTop + childParams.topMargin;
9609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (childParams.alignWithParent && rules[ALIGN_TOP] != 0) {
9619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mTop = mPaddingTop + childParams.topMargin;
9629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        anchorParams = getRelatedViewParams(rules, ALIGN_BOTTOM);
9659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (anchorParams != null) {
9669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mBottom = anchorParams.mBottom - childParams.bottomMargin;
9679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        } else if (childParams.alignWithParent && rules[ALIGN_BOTTOM] != 0) {
9689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (myHeight >= 0) {
9699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                childParams.mBottom = myHeight - mPaddingBottom - childParams.bottomMargin;
9709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
9719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9731b7c791adf6d5e113cd07fa1a9cbef310f79145aRomain Guy        if (0 != rules[ALIGN_PARENT_TOP]) {
9749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            childParams.mTop = mPaddingTop + childParams.topMargin;
9759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9771b7c791adf6d5e113cd07fa1a9cbef310f79145aRomain Guy        if (0 != rules[ALIGN_PARENT_BOTTOM]) {
9789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (myHeight >= 0) {
9799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                childParams.mBottom = myHeight - mPaddingBottom - childParams.bottomMargin;
9809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
9819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (rules[ALIGN_BASELINE] != 0) {
9849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mHasBaselineAlignedChild = true;
9859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
9869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private View getRelatedView(int[] rules, int relation) {
9899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int id = rules[relation];
9909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (id != 0) {
9911ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy            DependencyGraph.Node node = mGraph.mKeyNodes.get(id);
992a0fd1d742d8edaf6c7e79bdd16a9b0c44fda4503Romain Guy            if (node == null) return null;
993a0fd1d742d8edaf6c7e79bdd16a9b0c44fda4503Romain Guy            View v = node.view;
9949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            // Find the first non-GONE view up the chain
9969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            while (v.getVisibility() == View.GONE) {
99713705eda8cc065ee98feb94af649615579cd4a58Fabrice Di Meglio                rules = ((LayoutParams) v.getLayoutParams()).getRules(v.getLayoutDirection());
9981ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy                node = mGraph.mKeyNodes.get((rules[relation]));
999a0fd1d742d8edaf6c7e79bdd16a9b0c44fda4503Romain Guy                if (node == null) return null;
1000a0fd1d742d8edaf6c7e79bdd16a9b0c44fda4503Romain Guy                v = node.view;
10019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
10029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return v;
10049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return null;
10079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private LayoutParams getRelatedViewParams(int[] rules, int relation) {
10109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        View v = getRelatedView(rules, relation);
10119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (v != null) {
10129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            ViewGroup.LayoutParams params = v.getLayoutParams();
10139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (params instanceof LayoutParams) {
10149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                return (LayoutParams) v.getLayoutParams();
10159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
10169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return null;
10189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    private int getRelatedViewBaseline(int[] rules, int relation) {
10219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        View v = getRelatedView(rules, relation);
10229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (v != null) {
10239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return v.getBaseline();
10249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return -1;
10269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1028f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy    private static void centerHorizontal(View child, LayoutParams params, int myWidth) {
10299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int childWidth = child.getMeasuredWidth();
10309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int left = (myWidth - childWidth) / 2;
10319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        params.mLeft = left;
10339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        params.mRight = left + childWidth;
10349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1036f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy    private static void centerVertical(View child, LayoutParams params, int myHeight) {
10379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int childHeight = child.getMeasuredHeight();
10389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int top = (myHeight - childHeight) / 2;
10399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        params.mTop = top;
10419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        params.mBottom = top + childHeight;
10429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
10459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected void onLayout(boolean changed, int l, int t, int r, int b) {
10469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        //  The layout has actually already been performed and the positions
10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        //  cached.  Apply the cached values to the children.
104813705eda8cc065ee98feb94af649615579cd4a58Fabrice Di Meglio        final int count = getChildCount();
10499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        for (int i = 0; i < count; i++) {
10519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            View child = getChildAt(i);
10529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            if (child.getVisibility() != GONE) {
10539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                RelativeLayout.LayoutParams st =
10549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        (RelativeLayout.LayoutParams) child.getLayoutParams();
10559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                child.layout(st.mLeft, st.mTop, st.mRight, st.mBottom);
10569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
10579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
10589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
10619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public LayoutParams generateLayoutParams(AttributeSet attrs) {
10629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return new RelativeLayout.LayoutParams(getContext(), attrs);
10639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
10669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Returns a set of layout parameters with a width of
10679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * {@link android.view.ViewGroup.LayoutParams#WRAP_CONTENT},
10689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * a height of {@link android.view.ViewGroup.LayoutParams#WRAP_CONTENT} and no spanning.
10699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
10709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
10719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected ViewGroup.LayoutParams generateDefaultLayoutParams() {
10729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
10739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    // Override to allow type-checking of LayoutParams.
10769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
10779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected boolean checkLayoutParams(ViewGroup.LayoutParams p) {
10789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return p instanceof RelativeLayout.LayoutParams;
10799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    @Override
10829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    protected ViewGroup.LayoutParams generateLayoutParams(ViewGroup.LayoutParams p) {
10839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return new LayoutParams(p);
10849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
10859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
108675986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov    @Override
108775986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov    public boolean dispatchPopulateAccessibilityEvent(AccessibilityEvent event) {
108875986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        if (mTopToBottomLeftToRightSet == null) {
108975986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            mTopToBottomLeftToRightSet = new TreeSet<View>(new TopToBottomLeftToRightComparator());
109075986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        }
109175986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov
109275986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        // sort children top-to-bottom and left-to-right
109375986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        for (int i = 0, count = getChildCount(); i < count; i++) {
109475986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            mTopToBottomLeftToRightSet.add(getChildAt(i));
109575986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        }
109675986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov
109775986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        for (View view : mTopToBottomLeftToRightSet) {
10980b0a41d8e26eaf0f1d9d922621494daf40964a9aSvetoslav Ganov            if (view.getVisibility() == View.VISIBLE
10990b0a41d8e26eaf0f1d9d922621494daf40964a9aSvetoslav Ganov                    && view.dispatchPopulateAccessibilityEvent(event)) {
110075986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov                mTopToBottomLeftToRightSet.clear();
110175986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov                return true;
110275986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            }
110375986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        }
110475986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov
110575986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        mTopToBottomLeftToRightSet.clear();
110675986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        return false;
110775986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov    }
110875986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov
11098a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov    @Override
11108a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov    public void onInitializeAccessibilityEvent(AccessibilityEvent event) {
11118a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov        super.onInitializeAccessibilityEvent(event);
11128a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov        event.setClassName(RelativeLayout.class.getName());
11138a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov    }
11148a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov
11158a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov    @Override
11168a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov    public void onInitializeAccessibilityNodeInfo(AccessibilityNodeInfo info) {
11178a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov        super.onInitializeAccessibilityNodeInfo(info);
11188a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov        info.setClassName(RelativeLayout.class.getName());
11198a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov    }
11208a78fd4d9572dff95432fcc4ba0e87563415b728Svetoslav Ganov
112175986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov    /**
112275986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov     * Compares two views in left-to-right and top-to-bottom fashion.
112375986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov     */
112475986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov     private class TopToBottomLeftToRightComparator implements Comparator<View> {
112575986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        public int compare(View first, View second) {
112675986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            // top - bottom
112775986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            int topDifference = first.getTop() - second.getTop();
112875986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            if (topDifference != 0) {
112975986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov                return topDifference;
113075986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            }
113175986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            // left - right
113275986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            int leftDifference = first.getLeft() - second.getLeft();
113375986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            if (leftDifference != 0) {
113475986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov                return leftDifference;
113575986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            }
113675986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            // break tie by height
113775986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            int heightDiference = first.getHeight() - second.getHeight();
113875986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            if (heightDiference != 0) {
113975986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov                return heightDiference;
114075986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            }
114175986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            // break tie by width
114275986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            int widthDiference = first.getWidth() - second.getWidth();
114375986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            if (widthDiference != 0) {
114475986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov                return widthDiference;
114575986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            }
114675986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov            return 0;
114775986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov        }
114875986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov    }
114975986cf9bc57ef11ad70f36fb77fbbf5d63af6ecsvetoslavganov
11509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    /**
11519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * Per-child layout information associated with RelativeLayout.
11529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     *
11539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignWithParentIfMissing
11549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_toLeftOf
11559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_toRightOf
11569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_above
11579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_below
11589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignBaseline
11599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignLeft
11609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignTop
11619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignRight
11629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignBottom
11639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignParentLeft
11649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignParentTop
11659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignParentRight
11669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignParentBottom
11679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_centerInParent
11689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_centerHorizontal
11699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_centerVertical
1170f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_toStartOf
1171f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_toEndOf
1172f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignStart
1173f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignEnd
1174f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignParentStart
1175f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio     * @attr ref android.R.styleable#RelativeLayout_Layout_layout_alignParentEnd
11769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project     */
11779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    public static class LayoutParams extends ViewGroup.MarginLayoutParams {
1178bea95162ca25bd00b0479d93739b6283795c3986Konstantin Lopyrev        @ViewDebug.ExportedProperty(category = "layout", resolveId = true, indexMapping = {
1179c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = ABOVE,               to = "above"),
1180c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = ALIGN_BASELINE,      to = "alignBaseline"),
1181c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = ALIGN_BOTTOM,        to = "alignBottom"),
1182c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = ALIGN_LEFT,          to = "alignLeft"),
1183c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = ALIGN_PARENT_BOTTOM, to = "alignParentBottom"),
1184c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = ALIGN_PARENT_LEFT,   to = "alignParentLeft"),
1185c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = ALIGN_PARENT_RIGHT,  to = "alignParentRight"),
1186c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = ALIGN_PARENT_TOP,    to = "alignParentTop"),
1187c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = ALIGN_RIGHT,         to = "alignRight"),
1188c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = ALIGN_TOP,           to = "alignTop"),
1189c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = BELOW,               to = "below"),
1190c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = CENTER_HORIZONTAL,   to = "centerHorizontal"),
1191c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = CENTER_IN_PARENT,    to = "center"),
1192c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = CENTER_VERTICAL,     to = "centerVertical"),
1193c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project            @ViewDebug.IntToString(from = LEFT_OF,             to = "leftOf"),
1194f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            @ViewDebug.IntToString(from = RIGHT_OF,            to = "rightOf"),
1195f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            @ViewDebug.IntToString(from = ALIGN_START,         to = "alignStart"),
1196f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            @ViewDebug.IntToString(from = ALIGN_END,           to = "alignEnd"),
1197f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            @ViewDebug.IntToString(from = ALIGN_PARENT_START,  to = "alignParentStart"),
1198f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            @ViewDebug.IntToString(from = ALIGN_PARENT_END,    to = "alignParentEnd"),
1199f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            @ViewDebug.IntToString(from = START_OF,            to = "startOf"),
1200f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            @ViewDebug.IntToString(from = END_OF,              to = "endOf")
1201105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project        }, mapping = {
1202105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project            @ViewDebug.IntToString(from = TRUE, to = "true"),
1203a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy            @ViewDebug.IntToString(from = 0,    to = "false/NO_ID")
1204105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project        })
1205f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio
12069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        private int[] mRules = new int[VERB_COUNT];
1207f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        private int[] mInitialRules = new int[VERB_COUNT];
1208c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project
12099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        private int mLeft, mTop, mRight, mBottom;
12109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12110072f64939b37a4d84940656c2180ad2e0594ff4Fabrice Di Meglio        private int mStart = DEFAULT_MARGIN_RELATIVE;
12120072f64939b37a4d84940656c2180ad2e0594ff4Fabrice Di Meglio        private int mEnd = DEFAULT_MARGIN_RELATIVE;
1213f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio
1214f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        private boolean mRulesChanged = false;
1215c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        private boolean mIsRtlCompatibilityMode = false;
1216f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio
12179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
12189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * When true, uses the parent as the anchor if the anchor doesn't exist or if
12199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * the anchor's visibility is GONE.
12209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
1221bea95162ca25bd00b0479d93739b6283795c3986Konstantin Lopyrev        @ViewDebug.ExportedProperty(category = "layout")
12229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public boolean alignWithParent;
12239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public LayoutParams(Context c, AttributeSet attrs) {
12259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            super(c, attrs);
12269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            TypedArray a = c.obtainStyledAttributes(attrs,
12289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    com.android.internal.R.styleable.RelativeLayout_Layout);
12299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1230c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio            final int targetSdkVersion = c.getApplicationInfo().targetSdkVersion;
1231c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio            mIsRtlCompatibilityMode = (targetSdkVersion < JELLY_BEAN_MR1 ||
1232c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    !c.getApplicationInfo().hasRtlSupport());
1233c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio
12349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            final int[] rules = mRules;
1235f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy            //noinspection MismatchedReadAndWriteOfArray
1236f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            final int[] initialRules = mInitialRules;
12379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
12389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            final int N = a.getIndexCount();
12399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            for (int i = 0; i < N; i++) {
12409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                int attr = a.getIndex(i);
12419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                switch (attr) {
12429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignWithParentIfMissing:
12439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        alignWithParent = a.getBoolean(attr, false);
12449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_toLeftOf:
12469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[LEFT_OF] = a.getResourceId(attr, 0);
12479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_toRightOf:
12499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[RIGHT_OF] = a.getResourceId(attr, 0);
12509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_above:
12529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[ABOVE] = a.getResourceId(attr, 0);
12539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_below:
12559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[BELOW] = a.getResourceId(attr, 0);
12569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignBaseline:
12589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[ALIGN_BASELINE] = a.getResourceId(attr, 0);
12599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignLeft:
12619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[ALIGN_LEFT] = a.getResourceId(attr, 0);
12629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignTop:
12649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[ALIGN_TOP] = a.getResourceId(attr, 0);
12659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignRight:
12679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[ALIGN_RIGHT] = a.getResourceId(attr, 0);
12689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignBottom:
12709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[ALIGN_BOTTOM] = a.getResourceId(attr, 0);
12719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignParentLeft:
12739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[ALIGN_PARENT_LEFT] = a.getBoolean(attr, false) ? TRUE : 0;
12749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignParentTop:
12769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[ALIGN_PARENT_TOP] = a.getBoolean(attr, false) ? TRUE : 0;
12779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignParentRight:
12799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[ALIGN_PARENT_RIGHT] = a.getBoolean(attr, false) ? TRUE : 0;
12809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignParentBottom:
12829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[ALIGN_PARENT_BOTTOM] = a.getBoolean(attr, false) ? TRUE : 0;
12839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_centerInParent:
12859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[CENTER_IN_PARENT] = a.getBoolean(attr, false) ? TRUE : 0;
12869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_centerHorizontal:
12889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[CENTER_HORIZONTAL] = a.getBoolean(attr, false) ? TRUE : 0;
12899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        break;
12909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_centerVertical:
12919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                        rules[CENTER_VERTICAL] = a.getBoolean(attr, false) ? TRUE : 0;
12929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                       break;
1293f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_toStartOf:
1294f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        rules[START_OF] = a.getResourceId(attr, 0);
1295f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        break;
1296f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_toEndOf:
1297f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        rules[END_OF] = a.getResourceId(attr, 0);
1298f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        break;
1299f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignStart:
1300f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        rules[ALIGN_START] = a.getResourceId(attr, 0);
1301f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        break;
1302f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignEnd:
1303f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        rules[ALIGN_END] = a.getResourceId(attr, 0);
1304f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        break;
1305f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignParentStart:
1306f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        rules[ALIGN_PARENT_START] = a.getBoolean(attr, false) ? TRUE : 0;
1307f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        break;
1308f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    case com.android.internal.R.styleable.RelativeLayout_Layout_layout_alignParentEnd:
1309f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        rules[ALIGN_PARENT_END] = a.getBoolean(attr, false) ? TRUE : 0;
1310f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                        break;
13119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                }
13129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            }
13130072f64939b37a4d84940656c2180ad2e0594ff4Fabrice Di Meglio            mRulesChanged = true;
1314f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy            System.arraycopy(rules, LEFT_OF, initialRules, LEFT_OF, VERB_COUNT);
1315f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio
13169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            a.recycle();
13179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
13189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public LayoutParams(int w, int h) {
13209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            super(w, h);
13219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
13229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
13249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * {@inheritDoc}
13259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
13269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public LayoutParams(ViewGroup.LayoutParams source) {
13279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            super(source);
13289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
13299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
13319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * {@inheritDoc}
13329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
13339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public LayoutParams(ViewGroup.MarginLayoutParams source) {
13349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            super(source);
13359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
13369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13370a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette        /**
13380a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette         * Copy constructor. Clones the width, height, margin values, and rules
13390a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette         * of the source.
13400a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette         *
13410a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette         * @param source The layout params to copy from.
13420a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette         */
13430a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette        public LayoutParams(LayoutParams source) {
13440a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette            super(source);
13450a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette
13460a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette            this.mIsRtlCompatibilityMode = source.mIsRtlCompatibilityMode;
13470a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette            this.mRulesChanged = source.mRulesChanged;
13480a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette            this.alignWithParent = source.alignWithParent;
13490a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette
13500a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette            System.arraycopy(source.mRules, LEFT_OF, this.mRules, LEFT_OF, VERB_COUNT);
13510a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette            System.arraycopy(
13520a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette                    source.mInitialRules, LEFT_OF, this.mInitialRules, LEFT_OF, VERB_COUNT);
13530a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette        }
13540a0e155cadecd32599a7354a1836232c885f4bd2Alan Viverette
13559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        @Override
13569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public String debug(String output) {
13579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return output + "ViewGroup.LayoutParams={ width=" + sizeToString(width) +
13589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                    ", height=" + sizeToString(height) + " }";
13599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
13609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
13629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Adds a layout rule to be interpreted by the RelativeLayout. This
13639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * method should only be used for constraints that don't refer to another sibling
13649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * (e.g., CENTER_IN_PARENT) or take a boolean value ({@link RelativeLayout#TRUE}
1365f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         * for true or 0 for false). To specify a verb that takes a subject, use
13669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * {@link #addRule(int, int)} instead.
13679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
13689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @param verb One of the verbs defined by
13699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *        {@link android.widget.RelativeLayout RelativeLayout}, such as
13709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *        ALIGN_WITH_PARENT_LEFT.
13719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @see #addRule(int, int)
13729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
13739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void addRule(int verb) {
13749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mRules[verb] = TRUE;
1375f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            mInitialRules[verb] = TRUE;
1376f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            mRulesChanged = true;
13779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
13789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
13799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
13809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Adds a layout rule to be interpreted by the RelativeLayout. Use this for
13819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * verbs that take a target, such as a sibling (ALIGN_RIGHT) or a boolean
13829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * value (VISIBLE).
13839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
13849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @param verb One of the verbs defined by
13859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *        {@link android.widget.RelativeLayout RelativeLayout}, such as
13869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *         ALIGN_WITH_PARENT_LEFT.
13879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @param anchor The id of another view to use as an anchor,
13889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *        or a boolean value(represented as {@link RelativeLayout#TRUE})
13899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *        for true or 0 for false).  For verbs that don't refer to another sibling
13909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *        (for example, ALIGN_WITH_PARENT_BOTTOM) just use -1.
13919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @see #addRule(int)
13929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
13939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public void addRule(int verb, int anchor) {
13949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            mRules[verb] = anchor;
1395f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            mInitialRules[verb] = anchor;
1396f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            mRulesChanged = true;
1397f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        }
1398f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio
1399a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio        /**
1400a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio         * Removes a layout rule to be interpreted by the RelativeLayout.
1401a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio         *
1402a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio         * @param verb One of the verbs defined by
1403a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio         *        {@link android.widget.RelativeLayout RelativeLayout}, such as
1404a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio         *         ALIGN_WITH_PARENT_LEFT.
1405a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio         * @see #addRule(int)
1406a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio         * @see #addRule(int, int)
1407a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio         */
1408a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio        public void removeRule(int verb) {
1409a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio            mRules[verb] = 0;
1410a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio            mInitialRules[verb] = 0;
1411a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio            mRulesChanged = true;
1412a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio        }
1413a4c1030870d8c91b6d871922c3e6962323700f25Fabrice Di Meglio
1414f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        private boolean hasRelativeRules() {
1415f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            return (mInitialRules[START_OF] != 0 || mInitialRules[END_OF] != 0 ||
1416f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    mInitialRules[ALIGN_START] != 0 || mInitialRules[ALIGN_END] != 0 ||
1417f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    mInitialRules[ALIGN_PARENT_START] != 0 || mInitialRules[ALIGN_PARENT_END] != 0);
1418f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        }
1419f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio
1420c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        // The way we are resolving rules depends on the layout direction and if we are pre JB MR1
1421c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        // or not.
1422c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        //
1423c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        // If we are pre JB MR1 (said as "RTL compatibility mode"), "left"/"right" rules are having
1424c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        // predominance over any "start/end" rules that could have been defined. A special case:
1425c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        // if no "left"/"right" rule has been defined and "start"/"end" rules are defined then we
1426c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        // resolve those "start"/"end" rules to "left"/"right" respectively.
1427c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        //
1428c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        // If we are JB MR1+, then "start"/"end" rules are having predominance over "left"/"right"
1429c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        // rules. If no "start"/"end" rule is defined then we use "left"/"right" rules.
1430c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        //
1431c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        // In all cases, the result of the resolution should clear the "start"/"end" rules to leave
1432c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio        // only the "left"/"right" rules at the end.
1433f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        private void resolveRules(int layoutDirection) {
1434f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            final boolean isLayoutRtl = (layoutDirection == View.LAYOUT_DIRECTION_RTL);
1435c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio
1436f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            // Reset to initial state
1437f16c7a97548cedccea291ffebc4388a7f15c9aa1Romain Guy            System.arraycopy(mInitialRules, LEFT_OF, mRules, LEFT_OF, VERB_COUNT);
1438c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio
1439c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio            // Apply rules depending on direction and if we are in RTL compatibility mode
1440c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio            if (mIsRtlCompatibilityMode) {
1441c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[ALIGN_START] != 0) {
1442c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    if (mRules[ALIGN_LEFT] == 0) {
1443c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // "left" rule is not defined but "start" rule is: use the "start" rule as
1444c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // the "left" rule
1445c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        mRules[ALIGN_LEFT] = mRules[ALIGN_START];
1446c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    }
1447c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_START] = 0;
1448c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1449c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio
1450c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[ALIGN_END] != 0) {
1451c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    if (mRules[ALIGN_RIGHT] == 0) {
1452c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // "right" rule is not defined but "end" rule is: use the "end" rule as the
1453c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // "right" rule
1454c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        mRules[ALIGN_RIGHT] = mRules[ALIGN_END];
1455c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    }
1456c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_END] = 0;
1457c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1458c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio
1459c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[START_OF] != 0) {
1460c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    if (mRules[LEFT_OF] == 0) {
1461c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // "left" rule is not defined but "start" rule is: use the "start" rule as
1462c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // the "left" rule
1463c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        mRules[LEFT_OF] = mRules[START_OF];
1464c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    }
1465c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[START_OF] = 0;
1466c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1467c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio
1468c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[END_OF] != 0) {
1469c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    if (mRules[RIGHT_OF] == 0) {
1470c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // "right" rule is not defined but "end" rule is: use the "end" rule as the
1471c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // "right" rule
1472c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        mRules[RIGHT_OF] = mRules[END_OF];
1473c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    }
1474c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[END_OF] = 0;
1475c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1476c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio
1477c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[ALIGN_PARENT_START] != 0) {
1478c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    if (mRules[ALIGN_PARENT_LEFT] == 0) {
1479c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // "left" rule is not defined but "start" rule is: use the "start" rule as
1480c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // the "left" rule
1481c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        mRules[ALIGN_PARENT_LEFT] = mRules[ALIGN_PARENT_START];
1482c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    }
1483c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_PARENT_START] = 0;
1484c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1485c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio
1486c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[ALIGN_PARENT_RIGHT] == 0) {
1487c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    if (mRules[ALIGN_PARENT_RIGHT] == 0) {
1488c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // "right" rule is not defined but "end" rule is: use the "end" rule as the
1489c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        // "right" rule
1490c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        mRules[ALIGN_PARENT_RIGHT] = mRules[ALIGN_PARENT_END];
1491c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    }
1492c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_PARENT_END] = 0;
1493c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1494c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio            } else {
1495c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                // JB MR1+ case
1496c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if ((mRules[ALIGN_START] != 0 || mRules[ALIGN_END] != 0) &&
1497c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        (mRules[ALIGN_LEFT] != 0 || mRules[ALIGN_RIGHT] != 0)) {
1498c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    // "start"/"end" rules take precedence over "left"/"right" rules
1499c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_LEFT] = 0;
1500c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_RIGHT] = 0;
1501c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1502c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[ALIGN_START] != 0) {
1503c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    // "start" rule resolved to "left" or "right" depending on the direction
1504c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[isLayoutRtl ? ALIGN_RIGHT : ALIGN_LEFT] = mRules[ALIGN_START];
1505c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_START] = 0;
1506c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1507c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[ALIGN_END] != 0) {
1508c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    // "end" rule resolved to "left" or "right" depending on the direction
1509c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[isLayoutRtl ? ALIGN_LEFT : ALIGN_RIGHT] = mRules[ALIGN_END];
1510c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_END] = 0;
1511c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1512c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio
1513c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if ((mRules[START_OF] != 0 || mRules[END_OF] != 0) &&
1514c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        (mRules[LEFT_OF] != 0 || mRules[RIGHT_OF] != 0)) {
1515c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    // "start"/"end" rules take precedence over "left"/"right" rules
1516c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[LEFT_OF] = 0;
1517c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[RIGHT_OF] = 0;
1518c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1519c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[START_OF] != 0) {
1520c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    // "start" rule resolved to "left" or "right" depending on the direction
1521c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[isLayoutRtl ? RIGHT_OF : LEFT_OF] = mRules[START_OF];
1522c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[START_OF] = 0;
1523c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1524c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[END_OF] != 0) {
1525c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    // "end" rule resolved to "left" or "right" depending on the direction
1526c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[isLayoutRtl ? LEFT_OF : RIGHT_OF] = mRules[END_OF];
1527c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[END_OF] = 0;
1528c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1529c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio
1530c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if ((mRules[ALIGN_PARENT_START] != 0 || mRules[ALIGN_PARENT_END] != 0) &&
1531c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                        (mRules[ALIGN_PARENT_LEFT] != 0 || mRules[ALIGN_PARENT_RIGHT] != 0)) {
1532c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    // "start"/"end" rules take precedence over "left"/"right" rules
1533c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_PARENT_LEFT] = 0;
1534c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_PARENT_RIGHT] = 0;
1535c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1536c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[ALIGN_PARENT_START] != 0) {
1537c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    // "start" rule resolved to "left" or "right" depending on the direction
1538c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[isLayoutRtl ? ALIGN_PARENT_RIGHT : ALIGN_PARENT_LEFT] = mRules[ALIGN_PARENT_START];
1539c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_PARENT_START] = 0;
1540c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1541c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                if (mRules[ALIGN_PARENT_END] != 0) {
1542c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    // "end" rule resolved to "left" or "right" depending on the direction
1543c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[isLayoutRtl ? ALIGN_PARENT_LEFT : ALIGN_PARENT_RIGHT] = mRules[ALIGN_PARENT_END];
1544c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                    mRules[ALIGN_PARENT_END] = 0;
1545c44d880c508ded633b61930e1964df70876ec42fFabrice Di Meglio                }
1546f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            }
1547f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            mRulesChanged = false;
1548f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        }
1549f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio
1550f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        /**
1551f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         * Retrieves a complete list of all supported rules, where the index is the rule
1552f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         * verb, and the element value is the value specified, or "false" if it was never
1553f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         * set. If there are relative rules defined (*_START / *_END), they will be resolved
1554f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         * depending on the layout direction.
1555f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         *
1556f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         * @param layoutDirection the direction of the layout.
1557f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         *                        Should be either {@link View#LAYOUT_DIRECTION_LTR}
1558f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         *                        or {@link View#LAYOUT_DIRECTION_RTL}
1559f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         * @return the supported rules
1560f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         * @see #addRule(int, int)
1561f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         *
1562f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         * @hide
1563f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         */
1564f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        public int[] getRules(int layoutDirection) {
1565f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            if (hasRelativeRules() &&
1566f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    (mRulesChanged || layoutDirection != getLayoutDirection())) {
1567f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                resolveRules(layoutDirection);
1568f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                if (layoutDirection != getLayoutDirection()) {
1569f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                    setLayoutDirection(layoutDirection);
1570f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                }
1571f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            }
1572f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            return mRules;
15739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
15749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
15759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        /**
15769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * Retrieves a complete list of all supported rules, where the index is the rule
15779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * verb, and the element value is the value specified, or "false" if it was never
1578f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio         * set. There will be no resolution of relative rules done.
15799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         *
15809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @return the supported rules
15819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         * @see #addRule(int, int)
15829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project         */
15839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        public int[] getRules() {
15849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return mRules;
15859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
1586f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio
1587f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        @Override
15882918ab6c3258639148b8a5c78a34483af195246eFabrice Di Meglio        public void resolveLayoutDirection(int layoutDirection) {
1589f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            final boolean isLayoutRtl = isLayoutRtl();
1590f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            if (isLayoutRtl) {
15910072f64939b37a4d84940656c2180ad2e0594ff4Fabrice Di Meglio                if (mStart != DEFAULT_MARGIN_RELATIVE) mRight = mStart;
15920072f64939b37a4d84940656c2180ad2e0594ff4Fabrice Di Meglio                if (mEnd != DEFAULT_MARGIN_RELATIVE) mLeft = mEnd;
1593f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            } else {
15940072f64939b37a4d84940656c2180ad2e0594ff4Fabrice Di Meglio                if (mStart != DEFAULT_MARGIN_RELATIVE) mLeft = mStart;
15950072f64939b37a4d84940656c2180ad2e0594ff4Fabrice Di Meglio                if (mEnd != DEFAULT_MARGIN_RELATIVE) mRight = mEnd;
1596f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            }
1597f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio
1598f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            if (hasRelativeRules() && layoutDirection != getLayoutDirection()) {
1599f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio                resolveRules(layoutDirection);
1600f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            }
1601f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio            // This will set the layout direction
16022918ab6c3258639148b8a5c78a34483af195246eFabrice Di Meglio            super.resolveLayoutDirection(layoutDirection);
1603f443f98e7f41badd8f5d6f7bf7d26432e79a88edFabrice Di Meglio        }
16049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1605725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1606725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    private static class DependencyGraph {
1607725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        /**
16081ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy         * List of all views in the graph.
16091ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy         */
16101ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy        private ArrayList<Node> mNodes = new ArrayList<Node>();
16111ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy
16121ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy        /**
1613725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * List of nodes in the graph. Each node is identified by its
1614725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * view id (see View#getId()).
1615725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         */
16161ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy        private SparseArray<Node> mKeyNodes = new SparseArray<Node>();
1617725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1618725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        /**
1619725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * Temporary data structure used to build the list of roots
1620725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * for this graph.
1621725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         */
1622bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy        private ArrayDeque<Node> mRoots = new ArrayDeque<Node>();
1623725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1624725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        /**
1625725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * Clears the graph.
1626725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         */
1627725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        void clear() {
16281ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy            final ArrayList<Node> nodes = mNodes;
1629725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            final int count = nodes.size();
1630725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1631725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            for (int i = 0; i < count; i++) {
16321ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy                nodes.get(i).release();
1633725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            }
1634725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            nodes.clear();
1635725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
16361ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy            mKeyNodes.clear();
1637725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            mRoots.clear();
1638725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        }
1639725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1640725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        /**
1641725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * Adds a view to the graph.
1642725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         *
1643725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * @param view The view to be added as a node to the graph.
1644725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         */
1645725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        void add(View view) {
16461ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy            final int id = view.getId();
16471ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy            final Node node = Node.acquire(view);
16481ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy
16491ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy            if (id != View.NO_ID) {
16501ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy                mKeyNodes.put(id, node);
16511ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy            }
16521ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy
16531ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy            mNodes.add(node);
1654725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        }
1655725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1656725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        /**
1657725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * Builds a sorted list of views. The sorting order depends on the dependencies
1658725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * between the view. For instance, if view C needs view A to be processed first
1659725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * and view A needs view B to be processed first, the dependency graph
1660725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * is: B -> A -> C. The sorted array will contain views B, A and C in this order.
1661725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         *
1662725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * @param sorted The sorted list of views. The length of this array must
1663725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         *        be equal to getChildCount().
1664725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * @param rules The list of rules to take into account.
1665725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         */
1666725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        void getSortedViews(View[] sorted, int... rules) {
1667bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy            final ArrayDeque<Node> roots = findRoots(rules);
1668725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            int index = 0;
1669725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1670bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy            Node node;
1671bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy            while ((node = roots.pollLast()) != null) {
1672725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                final View view = node.view;
1673725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                final int key = view.getId();
1674725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1675725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                sorted[index++] = view;
1676725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
16776876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy                final ArrayMap<Node, DependencyGraph> dependents = node.dependents;
16786876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy                final int count = dependents.size();
16796876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy                for (int i = 0; i < count; i++) {
16806876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy                    final Node dependent = dependents.keyAt(i);
1681725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                    final SparseArray<Node> dependencies = dependent.dependencies;
1682725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1683725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                    dependencies.remove(key);
1684725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                    if (dependencies.size() == 0) {
1685725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                        roots.add(dependent);
1686725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                    }
1687725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                }
1688725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            }
1689725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1690725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            if (index < sorted.length) {
1691725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                throw new IllegalStateException("Circular dependencies cannot exist"
1692725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                        + " in RelativeLayout");
1693725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            }
1694725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        }
1695725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1696725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        /**
1697725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * Finds the roots of the graph. A root is a node with no dependency and
1698725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * with [0..n] dependents.
1699725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         *
1700725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * @param rulesFilter The list of rules to consider when building the
1701725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         *        dependencies
1702725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         *
1703725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * @return A list of node, each being a root of the graph
1704725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         */
1705bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy        private ArrayDeque<Node> findRoots(int[] rulesFilter) {
17061ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy            final SparseArray<Node> keyNodes = mKeyNodes;
17071ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy            final ArrayList<Node> nodes = mNodes;
1708725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            final int count = nodes.size();
1709725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1710725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            // Find roots can be invoked several times, so make sure to clear
1711725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            // all dependents and dependencies before running the algorithm
1712725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            for (int i = 0; i < count; i++) {
17131ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy                final Node node = nodes.get(i);
1714725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                node.dependents.clear();
1715725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                node.dependencies.clear();
1716725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            }
1717725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1718725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            // Builds up the dependents and dependencies for each node of the graph
1719725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            for (int i = 0; i < count; i++) {
17201ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy                final Node node = nodes.get(i);
1721725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1722725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                final LayoutParams layoutParams = (LayoutParams) node.view.getLayoutParams();
1723725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                final int[] rules = layoutParams.mRules;
1724725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                final int rulesCount = rulesFilter.length;
1725725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1726725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                // Look only the the rules passed in parameter, this way we build only the
1727725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                // dependencies for a specific set of rules
1728725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                for (int j = 0; j < rulesCount; j++) {
1729725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                    final int rule = rules[rulesFilter[j]];
1730725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                    if (rule > 0) {
1731725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                        // The node this node depends on
17321ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy                        final Node dependency = keyNodes.get(rule);
1733da3003e1d71d66a1c936489025f8db314a2a4588Romain Guy                        // Skip unknowns and self dependencies
1734da3003e1d71d66a1c936489025f8db314a2a4588Romain Guy                        if (dependency == null || dependency == node) {
1735b8f8de85160b0a072158b45320e9fc2adba545f5Romain Guy                            continue;
1736b8f8de85160b0a072158b45320e9fc2adba545f5Romain Guy                        }
1737725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                        // Add the current node as a dependent
1738bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy                        dependency.dependents.put(node, this);
1739725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                        // Add a dependency to the current node
1740725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                        node.dependencies.put(rule, dependency);
1741725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                    }
1742725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                }
1743725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            }
1744725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1745bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy            final ArrayDeque<Node> roots = mRoots;
1746725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            roots.clear();
1747725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1748725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            // Finds all the roots in the graph: all nodes with no dependencies
1749725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            for (int i = 0; i < count; i++) {
17501ab621e316828fa65e8941954e2a3c7f1d68f77aRomain Guy                final Node node = nodes.get(i);
1751bc5d876df0856e027f1e2cfce91cbdedb6aaf66fRomain Guy                if (node.dependencies.size() == 0) roots.addLast(node);
1752725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            }
1753725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1754725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            return roots;
1755725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        }
1756725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1757725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        /**
1758725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * A node in the dependency graph. A node is a view, its list of dependencies
1759725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * and its list of dependents.
1760725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         *
1761725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         * A node with no dependent is considered a root of the graph.
1762725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy         */
1763abae2a1b891772d36d8f781adfcc8969e551691fSvetoslav Ganov        static class Node {
1764725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            /**
1765725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             * The view representing this node in the layout.
1766725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             */
1767725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            View view;
1768725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1769725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            /**
1770725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             * The list of dependents for this node; a dependent is a node
1771725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             * that needs this node to be processed first.
1772725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             */
17736876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy            final ArrayMap<Node, DependencyGraph> dependents =
17746876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy                    new ArrayMap<Node, DependencyGraph>();
1775725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1776725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            /**
1777725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             * The list of dependencies for this node.
1778725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             */
1779725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            final SparseArray<Node> dependencies = new SparseArray<Node>();
1780725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1781725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            /*
1782725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             * START POOL IMPLEMENTATION
1783725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             */
1784baac46339da03aed166e8a4240ad063caad019adRomain Guy            // The pool is static, so all nodes instances are shared across
1785baac46339da03aed166e8a4240ad063caad019adRomain Guy            // activities, that's why we give it a rather high limit
1786baac46339da03aed166e8a4240ad063caad019adRomain Guy            private static final int POOL_LIMIT = 100;
17876876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy            private static final SynchronizedPool<Node> sPool =
17886876b4fb9082fe525d6edb59a834bb0e7c35ca39Romain Guy                    new SynchronizedPool<Node>(POOL_LIMIT);
17898643aa0179e598e78d938c59035389054535a229Svetoslav Ganov
1790725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            static Node acquire(View view) {
1791abae2a1b891772d36d8f781adfcc8969e551691fSvetoslav Ganov                Node node = sPool.acquire();
1792abae2a1b891772d36d8f781adfcc8969e551691fSvetoslav Ganov                if (node == null) {
1793abae2a1b891772d36d8f781adfcc8969e551691fSvetoslav Ganov                    node = new Node();
1794abae2a1b891772d36d8f781adfcc8969e551691fSvetoslav Ganov                }
1795725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                node.view = view;
1796725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                return node;
1797725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            }
1798725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1799725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            void release() {
1800725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                view = null;
1801725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                dependents.clear();
1802725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                dependencies.clear();
1803725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy
1804725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy                sPool.release(this);
1805725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            }
1806725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy            /*
1807725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             * END POOL IMPLEMENTATION
1808725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy             */
1809725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy        }
1810725015a9cda8f5bfcf05dff7d2b0ebbd799bb577Romain Guy    }
18119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1812