1/*
2 * Copyright (C) 2006 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 *      http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package android.widget;
18
19import com.android.internal.R;
20
21import android.content.Context;
22import android.content.res.TypedArray;
23import android.util.AttributeSet;
24import android.view.Gravity;
25import android.view.View;
26import android.view.ViewDebug;
27import android.view.ViewGroup;
28import android.widget.RemoteViews.RemoteView;
29
30
31/**
32 * A Layout that arranges its children in a single column or a single row. The direction of
33 * the row can be set by calling {@link #setOrientation(int) setOrientation()}.
34 * You can also specify gravity, which specifies the alignment of all the child elements by
35 * calling {@link #setGravity(int) setGravity()} or specify that specific children
36 * grow to fill up any remaining space in the layout by setting the <em>weight</em> member of
37 * {@link android.widget.LinearLayout.LayoutParams LinearLayout.LayoutParams}.
38 * The default orientation is horizontal.
39 *
40 * <p>See the <a href="{@docRoot}resources/tutorials/views/hello-linearlayout.html">Linear Layout
41 * tutorial</a>.</p>
42 *
43 * <p>
44 * Also see {@link LinearLayout.LayoutParams android.widget.LinearLayout.LayoutParams}
45 * for layout attributes </p>
46 */
47@RemoteView
48public class LinearLayout extends ViewGroup {
49    public static final int HORIZONTAL = 0;
50    public static final int VERTICAL = 1;
51
52    /**
53     * Whether the children of this layout are baseline aligned.  Only applicable
54     * if {@link #mOrientation} is horizontal.
55     */
56    @ViewDebug.ExportedProperty(category = "layout")
57    private boolean mBaselineAligned = true;
58
59    /**
60     * If this layout is part of another layout that is baseline aligned,
61     * use the child at this index as the baseline.
62     *
63     * Note: this is orthogonal to {@link #mBaselineAligned}, which is concerned
64     * with whether the children of this layout are baseline aligned.
65     */
66    @ViewDebug.ExportedProperty(category = "layout")
67    private int mBaselineAlignedChildIndex = -1;
68
69    /**
70     * The additional offset to the child's baseline.
71     * We'll calculate the baseline of this layout as we measure vertically; for
72     * horizontal linear layouts, the offset of 0 is appropriate.
73     */
74    @ViewDebug.ExportedProperty(category = "measurement")
75    private int mBaselineChildTop = 0;
76
77    @ViewDebug.ExportedProperty(category = "measurement")
78    private int mOrientation;
79
80    @ViewDebug.ExportedProperty(category = "measurement", mapping = {
81            @ViewDebug.IntToString(from =  -1,                       to = "NONE"),
82            @ViewDebug.IntToString(from = Gravity.NO_GRAVITY,        to = "NONE"),
83            @ViewDebug.IntToString(from = Gravity.TOP,               to = "TOP"),
84            @ViewDebug.IntToString(from = Gravity.BOTTOM,            to = "BOTTOM"),
85            @ViewDebug.IntToString(from = Gravity.LEFT,              to = "LEFT"),
86            @ViewDebug.IntToString(from = Gravity.RIGHT,             to = "RIGHT"),
87            @ViewDebug.IntToString(from = Gravity.CENTER_VERTICAL,   to = "CENTER_VERTICAL"),
88            @ViewDebug.IntToString(from = Gravity.FILL_VERTICAL,     to = "FILL_VERTICAL"),
89            @ViewDebug.IntToString(from = Gravity.CENTER_HORIZONTAL, to = "CENTER_HORIZONTAL"),
90            @ViewDebug.IntToString(from = Gravity.FILL_HORIZONTAL,   to = "FILL_HORIZONTAL"),
91            @ViewDebug.IntToString(from = Gravity.CENTER,            to = "CENTER"),
92            @ViewDebug.IntToString(from = Gravity.FILL,              to = "FILL")
93        })
94    private int mGravity = Gravity.LEFT | Gravity.TOP;
95
96    @ViewDebug.ExportedProperty(category = "measurement")
97    private int mTotalLength;
98
99    @ViewDebug.ExportedProperty(category = "layout")
100    private float mWeightSum;
101
102    @ViewDebug.ExportedProperty(category = "layout")
103    private boolean mUseLargestChild;
104
105    private int[] mMaxAscent;
106    private int[] mMaxDescent;
107
108    private static final int VERTICAL_GRAVITY_COUNT = 4;
109
110    private static final int INDEX_CENTER_VERTICAL = 0;
111    private static final int INDEX_TOP = 1;
112    private static final int INDEX_BOTTOM = 2;
113    private static final int INDEX_FILL = 3;
114
115    public LinearLayout(Context context) {
116        super(context);
117    }
118
119    public LinearLayout(Context context, AttributeSet attrs) {
120        super(context, attrs);
121
122        TypedArray a =
123            context.obtainStyledAttributes(attrs, com.android.internal.R.styleable.LinearLayout);
124
125        int index = a.getInt(com.android.internal.R.styleable.LinearLayout_orientation, -1);
126        if (index >= 0) {
127            setOrientation(index);
128        }
129
130        index = a.getInt(com.android.internal.R.styleable.LinearLayout_gravity, -1);
131        if (index >= 0) {
132            setGravity(index);
133        }
134
135        boolean baselineAligned = a.getBoolean(R.styleable.LinearLayout_baselineAligned, true);
136        if (!baselineAligned) {
137            setBaselineAligned(baselineAligned);
138        }
139
140        mWeightSum = a.getFloat(R.styleable.LinearLayout_weightSum, -1.0f);
141
142        mBaselineAlignedChildIndex =
143                a.getInt(com.android.internal.R.styleable.LinearLayout_baselineAlignedChildIndex, -1);
144
145        // TODO: Better name, add Java APIs, make it public
146        mUseLargestChild = a.getBoolean(R.styleable.LinearLayout_useLargestChild, false);
147
148        a.recycle();
149    }
150
151    /**
152     * <p>Indicates whether widgets contained within this layout are aligned
153     * on their baseline or not.</p>
154     *
155     * @return true when widgets are baseline-aligned, false otherwise
156     */
157    public boolean isBaselineAligned() {
158        return mBaselineAligned;
159    }
160
161    /**
162     * <p>Defines whether widgets contained in this layout are
163     * baseline-aligned or not.</p>
164     *
165     * @param baselineAligned true to align widgets on their baseline,
166     *         false otherwise
167     *
168     * @attr ref android.R.styleable#LinearLayout_baselineAligned
169     */
170    @android.view.RemotableViewMethod
171    public void setBaselineAligned(boolean baselineAligned) {
172        mBaselineAligned = baselineAligned;
173    }
174
175    @Override
176    public int getBaseline() {
177        if (mBaselineAlignedChildIndex < 0) {
178            return super.getBaseline();
179        }
180
181        if (getChildCount() <= mBaselineAlignedChildIndex) {
182            throw new RuntimeException("mBaselineAlignedChildIndex of LinearLayout "
183                    + "set to an index that is out of bounds.");
184        }
185
186        final View child = getChildAt(mBaselineAlignedChildIndex);
187        final int childBaseline = child.getBaseline();
188
189        if (childBaseline == -1) {
190            if (mBaselineAlignedChildIndex == 0) {
191                // this is just the default case, safe to return -1
192                return -1;
193            }
194            // the user picked an index that points to something that doesn't
195            // know how to calculate its baseline.
196            throw new RuntimeException("mBaselineAlignedChildIndex of LinearLayout "
197                    + "points to a View that doesn't know how to get its baseline.");
198        }
199
200        // TODO: This should try to take into account the virtual offsets
201        // (See getNextLocationOffset and getLocationOffset)
202        // We should add to childTop:
203        // sum([getNextLocationOffset(getChildAt(i)) / i < mBaselineAlignedChildIndex])
204        // and also add:
205        // getLocationOffset(child)
206        int childTop = mBaselineChildTop;
207
208        if (mOrientation == VERTICAL) {
209            final int majorGravity = mGravity & Gravity.VERTICAL_GRAVITY_MASK;
210            if (majorGravity != Gravity.TOP) {
211               switch (majorGravity) {
212                   case Gravity.BOTTOM:
213                       childTop = mBottom - mTop - mPaddingBottom - mTotalLength;
214                       break;
215
216                   case Gravity.CENTER_VERTICAL:
217                       childTop += ((mBottom - mTop - mPaddingTop - mPaddingBottom) -
218                               mTotalLength) / 2;
219                       break;
220               }
221            }
222        }
223
224        LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams) child.getLayoutParams();
225        return childTop + lp.topMargin + childBaseline;
226    }
227
228    /**
229     * @return The index of the child that will be used if this layout is
230     *   part of a larger layout that is baseline aligned, or -1 if none has
231     *   been set.
232     */
233    public int getBaselineAlignedChildIndex() {
234        return mBaselineAlignedChildIndex;
235    }
236
237    /**
238     * @param i The index of the child that will be used if this layout is
239     *          part of a larger layout that is baseline aligned.
240     *
241     * @attr ref android.R.styleable#LinearLayout_baselineAlignedChildIndex
242     */
243    @android.view.RemotableViewMethod
244    public void setBaselineAlignedChildIndex(int i) {
245        if ((i < 0) || (i >= getChildCount())) {
246            throw new IllegalArgumentException("base aligned child index out "
247                    + "of range (0, " + getChildCount() + ")");
248        }
249        mBaselineAlignedChildIndex = i;
250    }
251
252    /**
253     * <p>Returns the view at the specified index. This method can be overriden
254     * to take into account virtual children. Refer to
255     * {@link android.widget.TableLayout} and {@link android.widget.TableRow}
256     * for an example.</p>
257     *
258     * @param index the child's index
259     * @return the child at the specified index
260     */
261    View getVirtualChildAt(int index) {
262        return getChildAt(index);
263    }
264
265    /**
266     * <p>Returns the virtual number of children. This number might be different
267     * than the actual number of children if the layout can hold virtual
268     * children. Refer to
269     * {@link android.widget.TableLayout} and {@link android.widget.TableRow}
270     * for an example.</p>
271     *
272     * @return the virtual number of children
273     */
274    int getVirtualChildCount() {
275        return getChildCount();
276    }
277
278    /**
279     * Returns the desired weights sum.
280     *
281     * @return A number greater than 0.0f if the weight sum is defined, or
282     *         a number lower than or equals to 0.0f if not weight sum is
283     *         to be used.
284     */
285    public float getWeightSum() {
286        return mWeightSum;
287    }
288
289    /**
290     * Defines the desired weights sum. If unspecified the weights sum is computed
291     * at layout time by adding the layout_weight of each child.
292     *
293     * This can be used for instance to give a single child 50% of the total
294     * available space by giving it a layout_weight of 0.5 and setting the
295     * weightSum to 1.0.
296     *
297     * @param weightSum a number greater than 0.0f, or a number lower than or equals
298     *        to 0.0f if the weight sum should be computed from the children's
299     *        layout_weight
300     */
301    @android.view.RemotableViewMethod
302    public void setWeightSum(float weightSum) {
303        mWeightSum = Math.max(0.0f, weightSum);
304    }
305
306    @Override
307    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
308        if (mOrientation == VERTICAL) {
309            measureVertical(widthMeasureSpec, heightMeasureSpec);
310        } else {
311            measureHorizontal(widthMeasureSpec, heightMeasureSpec);
312        }
313    }
314
315    /**
316     * Measures the children when the orientation of this LinearLayout is set
317     * to {@link #VERTICAL}.
318     *
319     * @param widthMeasureSpec Horizontal space requirements as imposed by the parent.
320     * @param heightMeasureSpec Vertical space requirements as imposed by the parent.
321     *
322     * @see #getOrientation()
323     * @see #setOrientation(int)
324     * @see #onMeasure(int, int)
325     */
326    void measureVertical(int widthMeasureSpec, int heightMeasureSpec) {
327        mTotalLength = 0;
328        int maxWidth = 0;
329        int alternativeMaxWidth = 0;
330        int weightedMaxWidth = 0;
331        boolean allFillParent = true;
332        float totalWeight = 0;
333
334        final int count = getVirtualChildCount();
335
336        final int widthMode = MeasureSpec.getMode(widthMeasureSpec);
337        final int heightMode = MeasureSpec.getMode(heightMeasureSpec);
338
339        boolean matchWidth = false;
340
341        final int baselineChildIndex = mBaselineAlignedChildIndex;
342        final boolean useLargestChild = mUseLargestChild;
343
344        int largestChildHeight = Integer.MIN_VALUE;
345
346        // See how tall everyone is. Also remember max width.
347        for (int i = 0; i < count; ++i) {
348            final View child = getVirtualChildAt(i);
349
350            if (child == null) {
351                mTotalLength += measureNullChild(i);
352                continue;
353            }
354
355            if (child.getVisibility() == View.GONE) {
356               i += getChildrenSkipCount(child, i);
357               continue;
358            }
359
360            LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams) child.getLayoutParams();
361
362            totalWeight += lp.weight;
363
364            if (heightMode == MeasureSpec.EXACTLY && lp.height == 0 && lp.weight > 0) {
365                // Optimization: don't bother measuring children who are going to use
366                // leftover space. These views will get measured again down below if
367                // there is any leftover space.
368                final int totalLength = mTotalLength;
369                mTotalLength = Math.max(totalLength, totalLength + lp.topMargin + lp.bottomMargin);
370            } else {
371                int oldHeight = Integer.MIN_VALUE;
372
373                if (lp.height == 0 && lp.weight > 0) {
374                    // heightMode is either UNSPECIFIED or AT_MOST, and this
375                    // child wanted to stretch to fill available space.
376                    // Translate that to WRAP_CONTENT so that it does not end up
377                    // with a height of 0
378                    oldHeight = 0;
379                    lp.height = LayoutParams.WRAP_CONTENT;
380                }
381
382                // Determine how big this child would like to be. If this or
383                // previous children have given a weight, then we allow it to
384                // use all available space (and we will shrink things later
385                // if needed).
386                measureChildBeforeLayout(
387                       child, i, widthMeasureSpec, 0, heightMeasureSpec,
388                       totalWeight == 0 ? mTotalLength : 0);
389
390                if (oldHeight != Integer.MIN_VALUE) {
391                   lp.height = oldHeight;
392                }
393
394                final int childHeight = child.getMeasuredHeight();
395                final int totalLength = mTotalLength;
396                mTotalLength = Math.max(totalLength, totalLength + childHeight + lp.topMargin +
397                       lp.bottomMargin + getNextLocationOffset(child));
398
399                if (useLargestChild) {
400                    largestChildHeight = Math.max(childHeight, largestChildHeight);
401                }
402            }
403
404            /**
405             * If applicable, compute the additional offset to the child's baseline
406             * we'll need later when asked {@link #getBaseline}.
407             */
408            if ((baselineChildIndex >= 0) && (baselineChildIndex == i + 1)) {
409               mBaselineChildTop = mTotalLength;
410            }
411
412            // if we are trying to use a child index for our baseline, the above
413            // book keeping only works if there are no children above it with
414            // weight.  fail fast to aid the developer.
415            if (i < baselineChildIndex && lp.weight > 0) {
416                throw new RuntimeException("A child of LinearLayout with index "
417                        + "less than mBaselineAlignedChildIndex has weight > 0, which "
418                        + "won't work.  Either remove the weight, or don't set "
419                        + "mBaselineAlignedChildIndex.");
420            }
421
422            boolean matchWidthLocally = false;
423            if (widthMode != MeasureSpec.EXACTLY && lp.width == LayoutParams.MATCH_PARENT) {
424                // The width of the linear layout will scale, and at least one
425                // child said it wanted to match our width. Set a flag
426                // indicating that we need to remeasure at least that view when
427                // we know our width.
428                matchWidth = true;
429                matchWidthLocally = true;
430            }
431
432            final int margin = lp.leftMargin + lp.rightMargin;
433            final int measuredWidth = child.getMeasuredWidth() + margin;
434            maxWidth = Math.max(maxWidth, measuredWidth);
435
436            allFillParent = allFillParent && lp.width == LayoutParams.MATCH_PARENT;
437            if (lp.weight > 0) {
438                /*
439                 * Widths of weighted Views are bogus if we end up
440                 * remeasuring, so keep them separate.
441                 */
442                weightedMaxWidth = Math.max(weightedMaxWidth,
443                        matchWidthLocally ? margin : measuredWidth);
444            } else {
445                alternativeMaxWidth = Math.max(alternativeMaxWidth,
446                        matchWidthLocally ? margin : measuredWidth);
447            }
448
449            i += getChildrenSkipCount(child, i);
450        }
451
452        if (useLargestChild && heightMode == MeasureSpec.AT_MOST) {
453            mTotalLength = 0;
454
455            for (int i = 0; i < count; ++i) {
456                final View child = getVirtualChildAt(i);
457
458                if (child == null) {
459                    mTotalLength += measureNullChild(i);
460                    continue;
461                }
462
463                if (child.getVisibility() == GONE) {
464                    i += getChildrenSkipCount(child, i);
465                    continue;
466                }
467
468                final LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams)
469                        child.getLayoutParams();
470                // Account for negative margins
471                final int totalLength = mTotalLength;
472                mTotalLength = Math.max(totalLength, totalLength + largestChildHeight +
473                        lp.topMargin + lp.bottomMargin + getNextLocationOffset(child));
474            }
475        }
476
477        // Add in our padding
478        mTotalLength += mPaddingTop + mPaddingBottom;
479
480        int heightSize = mTotalLength;
481
482        // Check against our minimum height
483        heightSize = Math.max(heightSize, getSuggestedMinimumHeight());
484
485        // Reconcile our calculated size with the heightMeasureSpec
486        heightSize = resolveSize(heightSize, heightMeasureSpec);
487
488        // Either expand children with weight to take up available space or
489        // shrink them if they extend beyond our current bounds
490        int delta = heightSize - mTotalLength;
491        if (delta != 0 && totalWeight > 0.0f) {
492            float weightSum = mWeightSum > 0.0f ? mWeightSum : totalWeight;
493
494            mTotalLength = 0;
495
496            for (int i = 0; i < count; ++i) {
497                final View child = getVirtualChildAt(i);
498
499                if (child.getVisibility() == View.GONE) {
500                    continue;
501                }
502
503                LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams) child.getLayoutParams();
504
505                float childExtra = lp.weight;
506                if (childExtra > 0) {
507                    // Child said it could absorb extra space -- give him his share
508                    int share = (int) (childExtra * delta / weightSum);
509                    weightSum -= childExtra;
510                    delta -= share;
511
512                    final int childWidthMeasureSpec = getChildMeasureSpec(widthMeasureSpec,
513                            mPaddingLeft + mPaddingRight +
514                                    lp.leftMargin + lp.rightMargin, lp.width);
515
516                    // TODO: Use a field like lp.isMeasured to figure out if this
517                    // child has been previously measured
518                    if ((lp.height != 0) || (heightMode != MeasureSpec.EXACTLY)) {
519                        // child was measured once already above...
520                        // base new measurement on stored values
521                        int childHeight = child.getMeasuredHeight() + share;
522                        if (childHeight < 0) {
523                            childHeight = 0;
524                        }
525
526                        child.measure(childWidthMeasureSpec,
527                                MeasureSpec.makeMeasureSpec(childHeight, MeasureSpec.EXACTLY));
528                    } else {
529                        // child was skipped in the loop above.
530                        // Measure for this first time here
531                        child.measure(childWidthMeasureSpec,
532                                MeasureSpec.makeMeasureSpec(share > 0 ? share : 0,
533                                        MeasureSpec.EXACTLY));
534                    }
535                }
536
537                final int margin =  lp.leftMargin + lp.rightMargin;
538                final int measuredWidth = child.getMeasuredWidth() + margin;
539                maxWidth = Math.max(maxWidth, measuredWidth);
540
541                boolean matchWidthLocally = widthMode != MeasureSpec.EXACTLY &&
542                        lp.width == LayoutParams.MATCH_PARENT;
543
544                alternativeMaxWidth = Math.max(alternativeMaxWidth,
545                        matchWidthLocally ? margin : measuredWidth);
546
547                allFillParent = allFillParent && lp.width == LayoutParams.MATCH_PARENT;
548
549                final int totalLength = mTotalLength;
550                mTotalLength = Math.max(totalLength, totalLength + child.getMeasuredHeight() +
551                        lp.topMargin + lp.bottomMargin + getNextLocationOffset(child));
552            }
553
554            // Add in our padding
555            mTotalLength += mPaddingTop + mPaddingBottom;
556            // TODO: Should we recompute the heightSpec based on the new total length?
557        } else {
558            alternativeMaxWidth = Math.max(alternativeMaxWidth,
559                                           weightedMaxWidth);
560        }
561
562        if (!allFillParent && widthMode != MeasureSpec.EXACTLY) {
563            maxWidth = alternativeMaxWidth;
564        }
565
566        maxWidth += mPaddingLeft + mPaddingRight;
567
568        // Check against our minimum width
569        maxWidth = Math.max(maxWidth, getSuggestedMinimumWidth());
570
571        setMeasuredDimension(resolveSize(maxWidth, widthMeasureSpec), heightSize);
572
573        if (matchWidth) {
574            forceUniformWidth(count, heightMeasureSpec);
575        }
576    }
577
578    private void forceUniformWidth(int count, int heightMeasureSpec) {
579        // Pretend that the linear layout has an exact size.
580        int uniformMeasureSpec = MeasureSpec.makeMeasureSpec(getMeasuredWidth(),
581                MeasureSpec.EXACTLY);
582        for (int i = 0; i< count; ++i) {
583           final View child = getVirtualChildAt(i);
584           if (child.getVisibility() != GONE) {
585               LinearLayout.LayoutParams lp = ((LinearLayout.LayoutParams)child.getLayoutParams());
586
587               if (lp.width == LayoutParams.MATCH_PARENT) {
588                   // Temporarily force children to reuse their old measured height
589                   // FIXME: this may not be right for something like wrapping text?
590                   int oldHeight = lp.height;
591                   lp.height = child.getMeasuredHeight();
592
593                   // Remeasue with new dimensions
594                   measureChildWithMargins(child, uniformMeasureSpec, 0, heightMeasureSpec, 0);
595                   lp.height = oldHeight;
596               }
597           }
598        }
599    }
600
601    /**
602     * Measures the children when the orientation of this LinearLayout is set
603     * to {@link #HORIZONTAL}.
604     *
605     * @param widthMeasureSpec Horizontal space requirements as imposed by the parent.
606     * @param heightMeasureSpec Vertical space requirements as imposed by the parent.
607     *
608     * @see #getOrientation()
609     * @see #setOrientation(int)
610     * @see #onMeasure(int, int)
611     */
612    void measureHorizontal(int widthMeasureSpec, int heightMeasureSpec) {
613        mTotalLength = 0;
614        int maxHeight = 0;
615        int alternativeMaxHeight = 0;
616        int weightedMaxHeight = 0;
617        boolean allFillParent = true;
618        float totalWeight = 0;
619
620        final int count = getVirtualChildCount();
621
622        final int widthMode = MeasureSpec.getMode(widthMeasureSpec);
623        final int heightMode = MeasureSpec.getMode(heightMeasureSpec);
624
625        boolean matchHeight = false;
626
627        if (mMaxAscent == null || mMaxDescent == null) {
628            mMaxAscent = new int[VERTICAL_GRAVITY_COUNT];
629            mMaxDescent = new int[VERTICAL_GRAVITY_COUNT];
630        }
631
632        final int[] maxAscent = mMaxAscent;
633        final int[] maxDescent = mMaxDescent;
634
635        maxAscent[0] = maxAscent[1] = maxAscent[2] = maxAscent[3] = -1;
636        maxDescent[0] = maxDescent[1] = maxDescent[2] = maxDescent[3] = -1;
637
638        final boolean baselineAligned = mBaselineAligned;
639        final boolean useLargestChild = mUseLargestChild;
640
641        final boolean isExactly = widthMode == MeasureSpec.EXACTLY;
642
643        int largestChildWidth = Integer.MIN_VALUE;
644
645        // See how wide everyone is. Also remember max height.
646        for (int i = 0; i < count; ++i) {
647            final View child = getVirtualChildAt(i);
648
649            if (child == null) {
650                mTotalLength += measureNullChild(i);
651                continue;
652            }
653
654            if (child.getVisibility() == GONE) {
655                i += getChildrenSkipCount(child, i);
656                continue;
657            }
658
659            final LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams)
660                    child.getLayoutParams();
661
662            totalWeight += lp.weight;
663
664            if (widthMode == MeasureSpec.EXACTLY && lp.width == 0 && lp.weight > 0) {
665                // Optimization: don't bother measuring children who are going to use
666                // leftover space. These views will get measured again down below if
667                // there is any leftover space.
668                if (isExactly) {
669                    mTotalLength += lp.leftMargin + lp.rightMargin;
670                } else {
671                    final int totalLength = mTotalLength;
672                    mTotalLength = Math.max(totalLength, totalLength +
673                            lp.leftMargin + lp.rightMargin);
674                }
675
676                // Baseline alignment requires to measure widgets to obtain the
677                // baseline offset (in particular for TextViews). The following
678                // defeats the optimization mentioned above. Allow the child to
679                // use as much space as it wants because we can shrink things
680                // later (and re-measure).
681                if (baselineAligned) {
682                    final int freeSpec = MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED);
683                    child.measure(freeSpec, freeSpec);
684                }
685            } else {
686                int oldWidth = Integer.MIN_VALUE;
687
688                if (lp.width == 0 && lp.weight > 0) {
689                    // widthMode is either UNSPECIFIED or AT_MOST, and this
690                    // child
691                    // wanted to stretch to fill available space. Translate that to
692                    // WRAP_CONTENT so that it does not end up with a width of 0
693                    oldWidth = 0;
694                    lp.width = LayoutParams.WRAP_CONTENT;
695                }
696
697                // Determine how big this child would like to be. If this or
698                // previous children have given a weight, then we allow it to
699                // use all available space (and we will shrink things later
700                // if needed).
701                measureChildBeforeLayout(child, i, widthMeasureSpec,
702                        totalWeight == 0 ? mTotalLength : 0,
703                        heightMeasureSpec, 0);
704
705                if (oldWidth != Integer.MIN_VALUE) {
706                    lp.width = oldWidth;
707                }
708
709                final int childWidth = child.getMeasuredWidth();
710                if (isExactly) {
711                    mTotalLength += childWidth + lp.leftMargin + lp.rightMargin +
712                            getNextLocationOffset(child);
713                } else {
714                    final int totalLength = mTotalLength;
715                    mTotalLength = Math.max(totalLength, totalLength + childWidth + lp.leftMargin +
716                           lp.rightMargin + getNextLocationOffset(child));
717                }
718
719                if (useLargestChild) {
720                    largestChildWidth = Math.max(childWidth, largestChildWidth);
721                }
722            }
723
724            boolean matchHeightLocally = false;
725            if (heightMode != MeasureSpec.EXACTLY && lp.height == LayoutParams.MATCH_PARENT) {
726                // The height of the linear layout will scale, and at least one
727                // child said it wanted to match our height. Set a flag indicating that
728                // we need to remeasure at least that view when we know our height.
729                matchHeight = true;
730                matchHeightLocally = true;
731            }
732
733            final int margin = lp.topMargin + lp.bottomMargin;
734            final int childHeight = child.getMeasuredHeight() + margin;
735
736            if (baselineAligned) {
737                final int childBaseline = child.getBaseline();
738                if (childBaseline != -1) {
739                    // Translates the child's vertical gravity into an index
740                    // in the range 0..VERTICAL_GRAVITY_COUNT
741                    final int gravity = (lp.gravity < 0 ? mGravity : lp.gravity)
742                            & Gravity.VERTICAL_GRAVITY_MASK;
743                    final int index = ((gravity >> Gravity.AXIS_Y_SHIFT)
744                            & ~Gravity.AXIS_SPECIFIED) >> 1;
745
746                    maxAscent[index] = Math.max(maxAscent[index], childBaseline);
747                    maxDescent[index] = Math.max(maxDescent[index], childHeight - childBaseline);
748                }
749            }
750
751            maxHeight = Math.max(maxHeight, childHeight);
752
753            allFillParent = allFillParent && lp.height == LayoutParams.MATCH_PARENT;
754            if (lp.weight > 0) {
755                /*
756                 * Heights of weighted Views are bogus if we end up
757                 * remeasuring, so keep them separate.
758                 */
759                weightedMaxHeight = Math.max(weightedMaxHeight,
760                        matchHeightLocally ? margin : childHeight);
761            } else {
762                alternativeMaxHeight = Math.max(alternativeMaxHeight,
763                        matchHeightLocally ? margin : childHeight);
764            }
765
766            i += getChildrenSkipCount(child, i);
767        }
768
769        // Check mMaxAscent[INDEX_TOP] first because it maps to Gravity.TOP,
770        // the most common case
771        if (maxAscent[INDEX_TOP] != -1 ||
772                maxAscent[INDEX_CENTER_VERTICAL] != -1 ||
773                maxAscent[INDEX_BOTTOM] != -1 ||
774                maxAscent[INDEX_FILL] != -1) {
775            final int ascent = Math.max(maxAscent[INDEX_FILL],
776                    Math.max(maxAscent[INDEX_CENTER_VERTICAL],
777                    Math.max(maxAscent[INDEX_TOP], maxAscent[INDEX_BOTTOM])));
778            final int descent = Math.max(maxDescent[INDEX_FILL],
779                    Math.max(maxDescent[INDEX_CENTER_VERTICAL],
780                    Math.max(maxDescent[INDEX_TOP], maxDescent[INDEX_BOTTOM])));
781            maxHeight = Math.max(maxHeight, ascent + descent);
782        }
783
784        if (useLargestChild && widthMode == MeasureSpec.AT_MOST) {
785            mTotalLength = 0;
786
787            for (int i = 0; i < count; ++i) {
788                final View child = getVirtualChildAt(i);
789
790                if (child == null) {
791                    mTotalLength += measureNullChild(i);
792                    continue;
793                }
794
795                if (child.getVisibility() == GONE) {
796                    i += getChildrenSkipCount(child, i);
797                    continue;
798                }
799
800                final LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams)
801                        child.getLayoutParams();
802                if (isExactly) {
803                    mTotalLength += largestChildWidth + lp.leftMargin + lp.rightMargin +
804                            getNextLocationOffset(child);
805                } else {
806                    final int totalLength = mTotalLength;
807                    mTotalLength = Math.max(totalLength, totalLength + largestChildWidth +
808                            lp.leftMargin + lp.rightMargin + getNextLocationOffset(child));
809                }
810            }
811        }
812
813        // Add in our padding
814        mTotalLength += mPaddingLeft + mPaddingRight;
815
816        int widthSize = mTotalLength;
817
818        // Check against our minimum width
819        widthSize = Math.max(widthSize, getSuggestedMinimumWidth());
820
821        // Reconcile our calculated size with the widthMeasureSpec
822        widthSize = resolveSize(widthSize, widthMeasureSpec);
823
824        // Either expand children with weight to take up available space or
825        // shrink them if they extend beyond our current bounds
826        int delta = widthSize - mTotalLength;
827        if (delta != 0 && totalWeight > 0.0f) {
828            float weightSum = mWeightSum > 0.0f ? mWeightSum : totalWeight;
829
830            maxAscent[0] = maxAscent[1] = maxAscent[2] = maxAscent[3] = -1;
831            maxDescent[0] = maxDescent[1] = maxDescent[2] = maxDescent[3] = -1;
832            maxHeight = -1;
833
834            mTotalLength = 0;
835
836            for (int i = 0; i < count; ++i) {
837                final View child = getVirtualChildAt(i);
838
839                if (child == null || child.getVisibility() == View.GONE) {
840                    continue;
841                }
842
843                final LinearLayout.LayoutParams lp =
844                        (LinearLayout.LayoutParams) child.getLayoutParams();
845
846                float childExtra = lp.weight;
847                if (childExtra > 0) {
848                    // Child said it could absorb extra space -- give him his share
849                    int share = (int) (childExtra * delta / weightSum);
850                    weightSum -= childExtra;
851                    delta -= share;
852
853                    final int childHeightMeasureSpec = getChildMeasureSpec(
854                            heightMeasureSpec,
855                            mPaddingTop + mPaddingBottom + lp.topMargin + lp.bottomMargin,
856                            lp.height);
857
858                    // TODO: Use a field like lp.isMeasured to figure out if this
859                    // child has been previously measured
860                    if ((lp.width != 0) || (widthMode != MeasureSpec.EXACTLY)) {
861                        // child was measured once already above ... base new measurement
862                        // on stored values
863                        int childWidth = child.getMeasuredWidth() + share;
864                        if (childWidth < 0) {
865                            childWidth = 0;
866                        }
867
868                        child.measure(
869                            MeasureSpec.makeMeasureSpec(childWidth, MeasureSpec.EXACTLY),
870                            childHeightMeasureSpec);
871                    } else {
872                        // child was skipped in the loop above. Measure for this first time here
873                        child.measure(MeasureSpec.makeMeasureSpec(
874                                share > 0 ? share : 0, MeasureSpec.EXACTLY),
875                                childHeightMeasureSpec);
876                    }
877                }
878
879                if (isExactly) {
880                    mTotalLength += child.getMeasuredWidth() + lp.leftMargin + lp.rightMargin +
881                            getNextLocationOffset(child);
882                } else {
883                    final int totalLength = mTotalLength;
884                    mTotalLength = Math.max(totalLength, totalLength + child.getMeasuredWidth() +
885                            lp.leftMargin + lp.rightMargin + getNextLocationOffset(child));
886                }
887
888                boolean matchHeightLocally = heightMode != MeasureSpec.EXACTLY &&
889                        lp.height == LayoutParams.MATCH_PARENT;
890
891                final int margin = lp.topMargin + lp .bottomMargin;
892                int childHeight = child.getMeasuredHeight() + margin;
893                maxHeight = Math.max(maxHeight, childHeight);
894                alternativeMaxHeight = Math.max(alternativeMaxHeight,
895                        matchHeightLocally ? margin : childHeight);
896
897                allFillParent = allFillParent && lp.height == LayoutParams.MATCH_PARENT;
898
899                if (baselineAligned) {
900                    final int childBaseline = child.getBaseline();
901                    if (childBaseline != -1) {
902                        // Translates the child's vertical gravity into an index in the range 0..2
903                        final int gravity = (lp.gravity < 0 ? mGravity : lp.gravity)
904                                & Gravity.VERTICAL_GRAVITY_MASK;
905                        final int index = ((gravity >> Gravity.AXIS_Y_SHIFT)
906                                & ~Gravity.AXIS_SPECIFIED) >> 1;
907
908                        maxAscent[index] = Math.max(maxAscent[index], childBaseline);
909                        maxDescent[index] = Math.max(maxDescent[index],
910                                childHeight - childBaseline);
911                    }
912                }
913            }
914
915            // Add in our padding
916            mTotalLength += mPaddingLeft + mPaddingRight;
917            // TODO: Should we update widthSize with the new total length?
918
919            // Check mMaxAscent[INDEX_TOP] first because it maps to Gravity.TOP,
920            // the most common case
921            if (maxAscent[INDEX_TOP] != -1 ||
922                    maxAscent[INDEX_CENTER_VERTICAL] != -1 ||
923                    maxAscent[INDEX_BOTTOM] != -1 ||
924                    maxAscent[INDEX_FILL] != -1) {
925                final int ascent = Math.max(maxAscent[INDEX_FILL],
926                        Math.max(maxAscent[INDEX_CENTER_VERTICAL],
927                        Math.max(maxAscent[INDEX_TOP], maxAscent[INDEX_BOTTOM])));
928                final int descent = Math.max(maxDescent[INDEX_FILL],
929                        Math.max(maxDescent[INDEX_CENTER_VERTICAL],
930                        Math.max(maxDescent[INDEX_TOP], maxDescent[INDEX_BOTTOM])));
931                maxHeight = Math.max(maxHeight, ascent + descent);
932            }
933        } else {
934            alternativeMaxHeight = Math.max(alternativeMaxHeight, weightedMaxHeight);
935        }
936
937        if (!allFillParent && heightMode != MeasureSpec.EXACTLY) {
938            maxHeight = alternativeMaxHeight;
939        }
940
941        maxHeight += mPaddingTop + mPaddingBottom;
942
943        // Check against our minimum height
944        maxHeight = Math.max(maxHeight, getSuggestedMinimumHeight());
945
946        setMeasuredDimension(widthSize, resolveSize(maxHeight, heightMeasureSpec));
947
948        if (matchHeight) {
949            forceUniformHeight(count, widthMeasureSpec);
950        }
951    }
952
953    private void forceUniformHeight(int count, int widthMeasureSpec) {
954        // Pretend that the linear layout has an exact size. This is the measured height of
955        // ourselves. The measured height should be the max height of the children, changed
956        // to accomodate the heightMesureSpec from the parent
957        int uniformMeasureSpec = MeasureSpec.makeMeasureSpec(getMeasuredHeight(),
958                MeasureSpec.EXACTLY);
959        for (int i = 0; i < count; ++i) {
960           final View child = getVirtualChildAt(i);
961           if (child.getVisibility() != GONE) {
962               LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams) child.getLayoutParams();
963
964               if (lp.height == LayoutParams.MATCH_PARENT) {
965                   // Temporarily force children to reuse their old measured width
966                   // FIXME: this may not be right for something like wrapping text?
967                   int oldWidth = lp.width;
968                   lp.width = child.getMeasuredWidth();
969
970                   // Remeasure with new dimensions
971                   measureChildWithMargins(child, widthMeasureSpec, 0, uniformMeasureSpec, 0);
972                   lp.width = oldWidth;
973               }
974           }
975        }
976    }
977
978    /**
979     * <p>Returns the number of children to skip after measuring/laying out
980     * the specified child.</p>
981     *
982     * @param child the child after which we want to skip children
983     * @param index the index of the child after which we want to skip children
984     * @return the number of children to skip, 0 by default
985     */
986    int getChildrenSkipCount(View child, int index) {
987        return 0;
988    }
989
990    /**
991     * <p>Returns the size (width or height) that should be occupied by a null
992     * child.</p>
993     *
994     * @param childIndex the index of the null child
995     * @return the width or height of the child depending on the orientation
996     */
997    int measureNullChild(int childIndex) {
998        return 0;
999    }
1000
1001    /**
1002     * <p>Measure the child according to the parent's measure specs. This
1003     * method should be overriden by subclasses to force the sizing of
1004     * children. This method is called by {@link #measureVertical(int, int)} and
1005     * {@link #measureHorizontal(int, int)}.</p>
1006     *
1007     * @param child the child to measure
1008     * @param childIndex the index of the child in this view
1009     * @param widthMeasureSpec horizontal space requirements as imposed by the parent
1010     * @param totalWidth extra space that has been used up by the parent horizontally
1011     * @param heightMeasureSpec vertical space requirements as imposed by the parent
1012     * @param totalHeight extra space that has been used up by the parent vertically
1013     */
1014    void measureChildBeforeLayout(View child, int childIndex,
1015            int widthMeasureSpec, int totalWidth, int heightMeasureSpec,
1016            int totalHeight) {
1017        measureChildWithMargins(child, widthMeasureSpec, totalWidth,
1018                heightMeasureSpec, totalHeight);
1019    }
1020
1021    /**
1022     * <p>Return the location offset of the specified child. This can be used
1023     * by subclasses to change the location of a given widget.</p>
1024     *
1025     * @param child the child for which to obtain the location offset
1026     * @return the location offset in pixels
1027     */
1028    int getLocationOffset(View child) {
1029        return 0;
1030    }
1031
1032    /**
1033     * <p>Return the size offset of the next sibling of the specified child.
1034     * This can be used by subclasses to change the location of the widget
1035     * following <code>child</code>.</p>
1036     *
1037     * @param child the child whose next sibling will be moved
1038     * @return the location offset of the next child in pixels
1039     */
1040    int getNextLocationOffset(View child) {
1041        return 0;
1042    }
1043
1044    @Override
1045    protected void onLayout(boolean changed, int l, int t, int r, int b) {
1046        if (mOrientation == VERTICAL) {
1047            layoutVertical();
1048        } else {
1049            layoutHorizontal();
1050        }
1051    }
1052
1053    /**
1054     * Position the children during a layout pass if the orientation of this
1055     * LinearLayout is set to {@link #VERTICAL}.
1056     *
1057     * @see #getOrientation()
1058     * @see #setOrientation(int)
1059     * @see #onLayout(boolean, int, int, int, int)
1060     */
1061    void layoutVertical() {
1062        final int paddingLeft = mPaddingLeft;
1063
1064        int childTop = mPaddingTop;
1065        int childLeft;
1066
1067        // Where right end of child should go
1068        final int width = mRight - mLeft;
1069        int childRight = width - mPaddingRight;
1070
1071        // Space available for child
1072        int childSpace = width - paddingLeft - mPaddingRight;
1073
1074        final int count = getVirtualChildCount();
1075
1076        final int majorGravity = mGravity & Gravity.VERTICAL_GRAVITY_MASK;
1077        final int minorGravity = mGravity & Gravity.HORIZONTAL_GRAVITY_MASK;
1078
1079        if (majorGravity != Gravity.TOP) {
1080           switch (majorGravity) {
1081               case Gravity.BOTTOM:
1082                   // mTotalLength contains the padding already, we add the top
1083                   // padding to compensate
1084                   childTop = mBottom - mTop + mPaddingTop - mTotalLength;
1085                   break;
1086
1087               case Gravity.CENTER_VERTICAL:
1088                   childTop += ((mBottom - mTop)  - mTotalLength) / 2;
1089                   break;
1090           }
1091
1092        }
1093
1094        for (int i = 0; i < count; i++) {
1095            final View child = getVirtualChildAt(i);
1096            if (child == null) {
1097                childTop += measureNullChild(i);
1098            } else if (child.getVisibility() != GONE) {
1099                final int childWidth = child.getMeasuredWidth();
1100                final int childHeight = child.getMeasuredHeight();
1101
1102                final LinearLayout.LayoutParams lp =
1103                        (LinearLayout.LayoutParams) child.getLayoutParams();
1104
1105                int gravity = lp.gravity;
1106                if (gravity < 0) {
1107                    gravity = minorGravity;
1108                }
1109
1110                switch (gravity & Gravity.HORIZONTAL_GRAVITY_MASK) {
1111                    case Gravity.LEFT:
1112                        childLeft = paddingLeft + lp.leftMargin;
1113                        break;
1114
1115                    case Gravity.CENTER_HORIZONTAL:
1116                        childLeft = paddingLeft + ((childSpace - childWidth) / 2)
1117                                + lp.leftMargin - lp.rightMargin;
1118                        break;
1119
1120                    case Gravity.RIGHT:
1121                        childLeft = childRight - childWidth - lp.rightMargin;
1122                        break;
1123                    default:
1124                        childLeft = paddingLeft;
1125                        break;
1126                }
1127
1128
1129                childTop += lp.topMargin;
1130                setChildFrame(child, childLeft, childTop + getLocationOffset(child),
1131                        childWidth, childHeight);
1132                childTop += childHeight + lp.bottomMargin + getNextLocationOffset(child);
1133
1134                i += getChildrenSkipCount(child, i);
1135            }
1136        }
1137    }
1138
1139    /**
1140     * Position the children during a layout pass if the orientation of this
1141     * LinearLayout is set to {@link #HORIZONTAL}.
1142     *
1143     * @see #getOrientation()
1144     * @see #setOrientation(int)
1145     * @see #onLayout(boolean, int, int, int, int)
1146     */
1147    void layoutHorizontal() {
1148        final int paddingTop = mPaddingTop;
1149
1150        int childTop;
1151        int childLeft = mPaddingLeft;
1152
1153        // Where bottom of child should go
1154        final int height = mBottom - mTop;
1155        int childBottom = height - mPaddingBottom;
1156
1157        // Space available for child
1158        int childSpace = height - paddingTop - mPaddingBottom;
1159
1160        final int count = getVirtualChildCount();
1161
1162        final int majorGravity = mGravity & Gravity.HORIZONTAL_GRAVITY_MASK;
1163        final int minorGravity = mGravity & Gravity.VERTICAL_GRAVITY_MASK;
1164
1165        final boolean baselineAligned = mBaselineAligned;
1166
1167        final int[] maxAscent = mMaxAscent;
1168        final int[] maxDescent = mMaxDescent;
1169
1170        if (majorGravity != Gravity.LEFT) {
1171            switch (majorGravity) {
1172                case Gravity.RIGHT:
1173                    // mTotalLength contains the padding already, we add the left
1174                    // padding to compensate
1175                    childLeft = mRight - mLeft + mPaddingLeft - mTotalLength;
1176                    break;
1177
1178                case Gravity.CENTER_HORIZONTAL:
1179                    childLeft += ((mRight - mLeft) - mTotalLength) / 2;
1180                    break;
1181            }
1182       }
1183
1184        for (int i = 0; i < count; i++) {
1185            final View child = getVirtualChildAt(i);
1186
1187            if (child == null) {
1188                childLeft += measureNullChild(i);
1189            } else if (child.getVisibility() != GONE) {
1190                final int childWidth = child.getMeasuredWidth();
1191                final int childHeight = child.getMeasuredHeight();
1192                int childBaseline = -1;
1193
1194                final LinearLayout.LayoutParams lp =
1195                        (LinearLayout.LayoutParams) child.getLayoutParams();
1196
1197                if (baselineAligned && lp.height != LayoutParams.MATCH_PARENT) {
1198                    childBaseline = child.getBaseline();
1199                }
1200
1201                int gravity = lp.gravity;
1202                if (gravity < 0) {
1203                    gravity = minorGravity;
1204                }
1205
1206                switch (gravity & Gravity.VERTICAL_GRAVITY_MASK) {
1207                    case Gravity.TOP:
1208                        childTop = paddingTop + lp.topMargin;
1209                        if (childBaseline != -1) {
1210                            childTop += maxAscent[INDEX_TOP] - childBaseline;
1211                        }
1212                        break;
1213
1214                    case Gravity.CENTER_VERTICAL:
1215                        // Removed support for baseline alignment when layout_gravity or
1216                        // gravity == center_vertical. See bug #1038483.
1217                        // Keep the code around if we need to re-enable this feature
1218                        // if (childBaseline != -1) {
1219                        //     // Align baselines vertically only if the child is smaller than us
1220                        //     if (childSpace - childHeight > 0) {
1221                        //         childTop = paddingTop + (childSpace / 2) - childBaseline;
1222                        //     } else {
1223                        //         childTop = paddingTop + (childSpace - childHeight) / 2;
1224                        //     }
1225                        // } else {
1226                        childTop = paddingTop + ((childSpace - childHeight) / 2)
1227                                + lp.topMargin - lp.bottomMargin;
1228                        break;
1229
1230                    case Gravity.BOTTOM:
1231                        childTop = childBottom - childHeight - lp.bottomMargin;
1232                        if (childBaseline != -1) {
1233                            int descent = child.getMeasuredHeight() - childBaseline;
1234                            childTop -= (maxDescent[INDEX_BOTTOM] - descent);
1235                        }
1236                        break;
1237                    default:
1238                        childTop = paddingTop;
1239                        break;
1240                }
1241
1242                childLeft += lp.leftMargin;
1243                setChildFrame(child, childLeft + getLocationOffset(child), childTop,
1244                        childWidth, childHeight);
1245                childLeft += childWidth + lp.rightMargin +
1246                        getNextLocationOffset(child);
1247
1248                i += getChildrenSkipCount(child, i);
1249            }
1250        }
1251    }
1252
1253    private void setChildFrame(View child, int left, int top, int width, int height) {
1254        child.layout(left, top, left + width, top + height);
1255    }
1256
1257    /**
1258     * Should the layout be a column or a row.
1259     * @param orientation Pass HORIZONTAL or VERTICAL. Default
1260     * value is HORIZONTAL.
1261     *
1262     * @attr ref android.R.styleable#LinearLayout_orientation
1263     */
1264    public void setOrientation(int orientation) {
1265        if (mOrientation != orientation) {
1266            mOrientation = orientation;
1267            requestLayout();
1268        }
1269    }
1270
1271    /**
1272     * Returns the current orientation.
1273     *
1274     * @return either {@link #HORIZONTAL} or {@link #VERTICAL}
1275     */
1276    public int getOrientation() {
1277        return mOrientation;
1278    }
1279
1280    /**
1281     * Describes how the child views are positioned. Defaults to GRAVITY_TOP. If
1282     * this layout has a VERTICAL orientation, this controls where all the child
1283     * views are placed if there is extra vertical space. If this layout has a
1284     * HORIZONTAL orientation, this controls the alignment of the children.
1285     *
1286     * @param gravity See {@link android.view.Gravity}
1287     *
1288     * @attr ref android.R.styleable#LinearLayout_gravity
1289     */
1290    @android.view.RemotableViewMethod
1291    public void setGravity(int gravity) {
1292        if (mGravity != gravity) {
1293            if ((gravity & Gravity.HORIZONTAL_GRAVITY_MASK) == 0) {
1294                gravity |= Gravity.LEFT;
1295            }
1296
1297            if ((gravity & Gravity.VERTICAL_GRAVITY_MASK) == 0) {
1298                gravity |= Gravity.TOP;
1299            }
1300
1301            mGravity = gravity;
1302            requestLayout();
1303        }
1304    }
1305
1306    @android.view.RemotableViewMethod
1307    public void setHorizontalGravity(int horizontalGravity) {
1308        final int gravity = horizontalGravity & Gravity.HORIZONTAL_GRAVITY_MASK;
1309        if ((mGravity & Gravity.HORIZONTAL_GRAVITY_MASK) != gravity) {
1310            mGravity = (mGravity & ~Gravity.HORIZONTAL_GRAVITY_MASK) | gravity;
1311            requestLayout();
1312        }
1313    }
1314
1315    @android.view.RemotableViewMethod
1316    public void setVerticalGravity(int verticalGravity) {
1317        final int gravity = verticalGravity & Gravity.VERTICAL_GRAVITY_MASK;
1318        if ((mGravity & Gravity.VERTICAL_GRAVITY_MASK) != gravity) {
1319            mGravity = (mGravity & ~Gravity.VERTICAL_GRAVITY_MASK) | gravity;
1320            requestLayout();
1321        }
1322    }
1323
1324    @Override
1325    public LayoutParams generateLayoutParams(AttributeSet attrs) {
1326        return new LinearLayout.LayoutParams(getContext(), attrs);
1327    }
1328
1329    /**
1330     * Returns a set of layout parameters with a width of
1331     * {@link android.view.ViewGroup.LayoutParams#MATCH_PARENT}
1332     * and a height of {@link android.view.ViewGroup.LayoutParams#WRAP_CONTENT}
1333     * when the layout's orientation is {@link #VERTICAL}. When the orientation is
1334     * {@link #HORIZONTAL}, the width is set to {@link LayoutParams#WRAP_CONTENT}
1335     * and the height to {@link LayoutParams#WRAP_CONTENT}.
1336     */
1337    @Override
1338    protected LayoutParams generateDefaultLayoutParams() {
1339        if (mOrientation == HORIZONTAL) {
1340            return new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
1341        } else if (mOrientation == VERTICAL) {
1342            return new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT);
1343        }
1344        return null;
1345    }
1346
1347    @Override
1348    protected LayoutParams generateLayoutParams(ViewGroup.LayoutParams p) {
1349        return new LayoutParams(p);
1350    }
1351
1352
1353    // Override to allow type-checking of LayoutParams.
1354    @Override
1355    protected boolean checkLayoutParams(ViewGroup.LayoutParams p) {
1356        return p instanceof LinearLayout.LayoutParams;
1357    }
1358
1359    /**
1360     * Per-child layout information associated with ViewLinearLayout.
1361     *
1362     * @attr ref android.R.styleable#LinearLayout_Layout_layout_weight
1363     * @attr ref android.R.styleable#LinearLayout_Layout_layout_gravity
1364     */
1365    public static class LayoutParams extends ViewGroup.MarginLayoutParams {
1366        /**
1367         * Indicates how much of the extra space in the LinearLayout will be
1368         * allocated to the view associated with these LayoutParams. Specify
1369         * 0 if the view should not be stretched. Otherwise the extra pixels
1370         * will be pro-rated among all views whose weight is greater than 0.
1371         */
1372        @ViewDebug.ExportedProperty(category = "layout")
1373        public float weight;
1374
1375        /**
1376         * Gravity for the view associated with these LayoutParams.
1377         *
1378         * @see android.view.Gravity
1379         */
1380        @ViewDebug.ExportedProperty(category = "layout", mapping = {
1381            @ViewDebug.IntToString(from =  -1,                       to = "NONE"),
1382            @ViewDebug.IntToString(from = Gravity.NO_GRAVITY,        to = "NONE"),
1383            @ViewDebug.IntToString(from = Gravity.TOP,               to = "TOP"),
1384            @ViewDebug.IntToString(from = Gravity.BOTTOM,            to = "BOTTOM"),
1385            @ViewDebug.IntToString(from = Gravity.LEFT,              to = "LEFT"),
1386            @ViewDebug.IntToString(from = Gravity.RIGHT,             to = "RIGHT"),
1387            @ViewDebug.IntToString(from = Gravity.CENTER_VERTICAL,   to = "CENTER_VERTICAL"),
1388            @ViewDebug.IntToString(from = Gravity.FILL_VERTICAL,     to = "FILL_VERTICAL"),
1389            @ViewDebug.IntToString(from = Gravity.CENTER_HORIZONTAL, to = "CENTER_HORIZONTAL"),
1390            @ViewDebug.IntToString(from = Gravity.FILL_HORIZONTAL,   to = "FILL_HORIZONTAL"),
1391            @ViewDebug.IntToString(from = Gravity.CENTER,            to = "CENTER"),
1392            @ViewDebug.IntToString(from = Gravity.FILL,              to = "FILL")
1393        })
1394        public int gravity = -1;
1395
1396        /**
1397         * {@inheritDoc}
1398         */
1399        public LayoutParams(Context c, AttributeSet attrs) {
1400            super(c, attrs);
1401            TypedArray a =
1402                    c.obtainStyledAttributes(attrs, com.android.internal.R.styleable.LinearLayout_Layout);
1403
1404            weight = a.getFloat(com.android.internal.R.styleable.LinearLayout_Layout_layout_weight, 0);
1405            gravity = a.getInt(com.android.internal.R.styleable.LinearLayout_Layout_layout_gravity, -1);
1406
1407            a.recycle();
1408        }
1409
1410        /**
1411         * {@inheritDoc}
1412         */
1413        public LayoutParams(int width, int height) {
1414            super(width, height);
1415            weight = 0;
1416        }
1417
1418        /**
1419         * Creates a new set of layout parameters with the specified width, height
1420         * and weight.
1421         *
1422         * @param width the width, either {@link #MATCH_PARENT},
1423         *        {@link #WRAP_CONTENT} or a fixed size in pixels
1424         * @param height the height, either {@link #MATCH_PARENT},
1425         *        {@link #WRAP_CONTENT} or a fixed size in pixels
1426         * @param weight the weight
1427         */
1428        public LayoutParams(int width, int height, float weight) {
1429            super(width, height);
1430            this.weight = weight;
1431        }
1432
1433        /**
1434         * {@inheritDoc}
1435         */
1436        public LayoutParams(ViewGroup.LayoutParams p) {
1437            super(p);
1438        }
1439
1440        /**
1441         * {@inheritDoc}
1442         */
1443        public LayoutParams(MarginLayoutParams source) {
1444            super(source);
1445        }
1446
1447        @Override
1448        public String debug(String output) {
1449            return output + "LinearLayout.LayoutParams={width=" + sizeToString(width) +
1450                    ", height=" + sizeToString(height) + " weight=" + weight +  "}";
1451        }
1452    }
1453}
1454