From 88254d822fa511da2681149f0ecbbe551b043af2 Mon Sep 17 00:00:00 2001
From: Julian Winkler
Date: Sun, 4 Feb 2024 16:15:26 +0100
Subject: [PATCH] replace LinearLayout with AOSPs implementation
Also update QuickHelp.md, as LinearLayout was the last custom
implemented ViewGroup
---
doc/QuickHelp.md | 10 +-
meson.build | 1 -
.../android_widget_LinearLayout.h | 237 ---
.../widgets/android_widget_LinearLayout.c | 44 -
src/api-impl/android/widget/LinearLayout.java | 1795 ++++++++++++++++-
5 files changed, 1774 insertions(+), 313 deletions(-)
delete mode 100644 src/api-impl-jni/generated_headers/android_widget_LinearLayout.h
delete mode 100644 src/api-impl-jni/widgets/android_widget_LinearLayout.c
diff --git a/doc/QuickHelp.md b/doc/QuickHelp.md
index 3722e022..3b9d2e30 100644
--- a/doc/QuickHelp.md
+++ b/doc/QuickHelp.md
@@ -93,13 +93,9 @@ to get to work in order to cut down on the amount of stubbing you need to do.
There are two basic types of widgets (Views): containers (Layouts) and the rest.
-To implement a container widget, simply copy an existing container widget implementation (e.g LinearLayout
-(`src/api-impl/android/widget/LinearLayout.java` and `src/api-impl-jni/widgets/android_widget_LinearLayout.c`)), and that's
-it! Now, chances are that you wanted something slightly different, but this will at least display the child
-widgets so that you can focus on implementing those.
-
-When you get around to properly implementing the particularities of the specific container widget, it's
-basically the same process as below.
+Initially all container widgets where backed by Gtk container widgets. As this caused lots of behaviour diffeneces with AOSP,
+we have instead implemented the API of ViewGroup, which is the super class of all container widgets. This allows to more or
+less completely reuse specialized container widget implementation from AOSP source code.
To implement any other widget, copy a widget that is closest to what you're looking for, and if Gtk has
a better approximation for your widget, then change to that as the backing Gtk widget. If Gtk doesn't have
diff --git a/meson.build b/meson.build
index 816a071e..82d4b146 100644
--- a/meson.build
+++ b/meson.build
@@ -98,7 +98,6 @@ libtranslationlayer_so = shared_library('translation_layer_main', [
'src/api-impl-jni/widgets/android_widget_ImageView.c',
'src/api-impl-jni/widgets/WrapperWidget.c',
'src/api-impl-jni/widgets/android_widget_TextView.c',
- 'src/api-impl-jni/widgets/android_widget_LinearLayout.c',
'src/api-impl-jni/widgets/android_widget_Progressbar.c',
'src/api-impl-jni/widgets/android_view_SurfaceView.c',
'src/api-impl-jni/views/AndroidLayout.c',
diff --git a/src/api-impl-jni/generated_headers/android_widget_LinearLayout.h b/src/api-impl-jni/generated_headers/android_widget_LinearLayout.h
deleted file mode 100644
index a217314b..00000000
--- a/src/api-impl-jni/generated_headers/android_widget_LinearLayout.h
+++ /dev/null
@@ -1,237 +0,0 @@
-/* DO NOT EDIT THIS FILE - it is machine generated */
-#include
-/* Header for class android_widget_LinearLayout */
-
-#ifndef _Included_android_widget_LinearLayout
-#define _Included_android_widget_LinearLayout
-#ifdef __cplusplus
-extern "C" {
-#endif
-#undef android_widget_LinearLayout_NO_ID
-#define android_widget_LinearLayout_NO_ID -1L
-#undef android_widget_LinearLayout_NOT_FOCUSABLE
-#define android_widget_LinearLayout_NOT_FOCUSABLE 0L
-#undef android_widget_LinearLayout_FOCUSABLE
-#define android_widget_LinearLayout_FOCUSABLE 1L
-#undef android_widget_LinearLayout_FOCUSABLE_MASK
-#define android_widget_LinearLayout_FOCUSABLE_MASK 1L
-#undef android_widget_LinearLayout_FITS_SYSTEM_WINDOWS
-#define android_widget_LinearLayout_FITS_SYSTEM_WINDOWS 2L
-#undef android_widget_LinearLayout_VISIBLE
-#define android_widget_LinearLayout_VISIBLE 0L
-#undef android_widget_LinearLayout_INVISIBLE
-#define android_widget_LinearLayout_INVISIBLE 4L
-#undef android_widget_LinearLayout_GONE
-#define android_widget_LinearLayout_GONE 8L
-#undef android_widget_LinearLayout_VISIBILITY_MASK
-#define android_widget_LinearLayout_VISIBILITY_MASK 12L
-#undef android_widget_LinearLayout_ENABLED
-#define android_widget_LinearLayout_ENABLED 0L
-#undef android_widget_LinearLayout_DISABLED
-#define android_widget_LinearLayout_DISABLED 32L
-#undef android_widget_LinearLayout_ENABLED_MASK
-#define android_widget_LinearLayout_ENABLED_MASK 32L
-#undef android_widget_LinearLayout_WILL_NOT_DRAW
-#define android_widget_LinearLayout_WILL_NOT_DRAW 128L
-#undef android_widget_LinearLayout_DRAW_MASK
-#define android_widget_LinearLayout_DRAW_MASK 128L
-#undef android_widget_LinearLayout_SCROLLBARS_NONE
-#define android_widget_LinearLayout_SCROLLBARS_NONE 0L
-#undef android_widget_LinearLayout_SCROLLBARS_HORIZONTAL
-#define android_widget_LinearLayout_SCROLLBARS_HORIZONTAL 256L
-#undef android_widget_LinearLayout_SCROLLBARS_VERTICAL
-#define android_widget_LinearLayout_SCROLLBARS_VERTICAL 512L
-#undef android_widget_LinearLayout_SCROLLBARS_MASK
-#define android_widget_LinearLayout_SCROLLBARS_MASK 768L
-#undef android_widget_LinearLayout_FILTER_TOUCHES_WHEN_OBSCURED
-#define android_widget_LinearLayout_FILTER_TOUCHES_WHEN_OBSCURED 1024L
-#undef android_widget_LinearLayout_OPTIONAL_FITS_SYSTEM_WINDOWS
-#define android_widget_LinearLayout_OPTIONAL_FITS_SYSTEM_WINDOWS 2048L
-#undef android_widget_LinearLayout_FADING_EDGE_NONE
-#define android_widget_LinearLayout_FADING_EDGE_NONE 0L
-#undef android_widget_LinearLayout_FADING_EDGE_HORIZONTAL
-#define android_widget_LinearLayout_FADING_EDGE_HORIZONTAL 4096L
-#undef android_widget_LinearLayout_FADING_EDGE_VERTICAL
-#define android_widget_LinearLayout_FADING_EDGE_VERTICAL 8192L
-#undef android_widget_LinearLayout_FADING_EDGE_MASK
-#define android_widget_LinearLayout_FADING_EDGE_MASK 12288L
-#undef android_widget_LinearLayout_CLICKABLE
-#define android_widget_LinearLayout_CLICKABLE 16384L
-#undef android_widget_LinearLayout_DRAWING_CACHE_ENABLED
-#define android_widget_LinearLayout_DRAWING_CACHE_ENABLED 32768L
-#undef android_widget_LinearLayout_SAVE_DISABLED
-#define android_widget_LinearLayout_SAVE_DISABLED 65536L
-#undef android_widget_LinearLayout_SAVE_DISABLED_MASK
-#define android_widget_LinearLayout_SAVE_DISABLED_MASK 65536L
-#undef android_widget_LinearLayout_WILL_NOT_CACHE_DRAWING
-#define android_widget_LinearLayout_WILL_NOT_CACHE_DRAWING 131072L
-#undef android_widget_LinearLayout_FOCUSABLE_IN_TOUCH_MODE
-#define android_widget_LinearLayout_FOCUSABLE_IN_TOUCH_MODE 262144L
-#undef android_widget_LinearLayout_DRAWING_CACHE_QUALITY_LOW
-#define android_widget_LinearLayout_DRAWING_CACHE_QUALITY_LOW 524288L
-#undef android_widget_LinearLayout_DRAWING_CACHE_QUALITY_HIGH
-#define android_widget_LinearLayout_DRAWING_CACHE_QUALITY_HIGH 1048576L
-#undef android_widget_LinearLayout_DRAWING_CACHE_QUALITY_AUTO
-#define android_widget_LinearLayout_DRAWING_CACHE_QUALITY_AUTO 0L
-#undef android_widget_LinearLayout_DRAWING_CACHE_QUALITY_MASK
-#define android_widget_LinearLayout_DRAWING_CACHE_QUALITY_MASK 1572864L
-#undef android_widget_LinearLayout_LONG_CLICKABLE
-#define android_widget_LinearLayout_LONG_CLICKABLE 2097152L
-#undef android_widget_LinearLayout_DUPLICATE_PARENT_STATE
-#define android_widget_LinearLayout_DUPLICATE_PARENT_STATE 4194304L
-#undef android_widget_LinearLayout_SCROLLBARS_INSIDE_OVERLAY
-#define android_widget_LinearLayout_SCROLLBARS_INSIDE_OVERLAY 0L
-#undef android_widget_LinearLayout_SCROLLBARS_INSIDE_INSET
-#define android_widget_LinearLayout_SCROLLBARS_INSIDE_INSET 16777216L
-#undef android_widget_LinearLayout_SCROLLBARS_OUTSIDE_OVERLAY
-#define android_widget_LinearLayout_SCROLLBARS_OUTSIDE_OVERLAY 33554432L
-#undef android_widget_LinearLayout_SCROLLBARS_OUTSIDE_INSET
-#define android_widget_LinearLayout_SCROLLBARS_OUTSIDE_INSET 50331648L
-#undef android_widget_LinearLayout_SCROLLBARS_INSET_MASK
-#define android_widget_LinearLayout_SCROLLBARS_INSET_MASK 16777216L
-#undef android_widget_LinearLayout_SCROLLBARS_OUTSIDE_MASK
-#define android_widget_LinearLayout_SCROLLBARS_OUTSIDE_MASK 33554432L
-#undef android_widget_LinearLayout_SCROLLBARS_STYLE_MASK
-#define android_widget_LinearLayout_SCROLLBARS_STYLE_MASK 50331648L
-#undef android_widget_LinearLayout_KEEP_SCREEN_ON
-#define android_widget_LinearLayout_KEEP_SCREEN_ON 67108864L
-#undef android_widget_LinearLayout_SOUND_EFFECTS_ENABLED
-#define android_widget_LinearLayout_SOUND_EFFECTS_ENABLED 134217728L
-#undef android_widget_LinearLayout_HAPTIC_FEEDBACK_ENABLED
-#define android_widget_LinearLayout_HAPTIC_FEEDBACK_ENABLED 268435456L
-#undef android_widget_LinearLayout_PARENT_SAVE_DISABLED
-#define android_widget_LinearLayout_PARENT_SAVE_DISABLED 536870912L
-#undef android_widget_LinearLayout_PARENT_SAVE_DISABLED_MASK
-#define android_widget_LinearLayout_PARENT_SAVE_DISABLED_MASK 536870912L
-#undef android_widget_LinearLayout_FOCUSABLES_ALL
-#define android_widget_LinearLayout_FOCUSABLES_ALL 0L
-#undef android_widget_LinearLayout_FOCUSABLES_TOUCH_MODE
-#define android_widget_LinearLayout_FOCUSABLES_TOUCH_MODE 1L
-#undef android_widget_LinearLayout_FOCUS_BACKWARD
-#define android_widget_LinearLayout_FOCUS_BACKWARD 1L
-#undef android_widget_LinearLayout_FOCUS_FORWARD
-#define android_widget_LinearLayout_FOCUS_FORWARD 2L
-#undef android_widget_LinearLayout_FOCUS_LEFT
-#define android_widget_LinearLayout_FOCUS_LEFT 17L
-#undef android_widget_LinearLayout_FOCUS_UP
-#define android_widget_LinearLayout_FOCUS_UP 33L
-#undef android_widget_LinearLayout_FOCUS_RIGHT
-#define android_widget_LinearLayout_FOCUS_RIGHT 66L
-#undef android_widget_LinearLayout_FOCUS_DOWN
-#define android_widget_LinearLayout_FOCUS_DOWN 130L
-#undef android_widget_LinearLayout_MEASURED_SIZE_MASK
-#define android_widget_LinearLayout_MEASURED_SIZE_MASK 16777215L
-#undef android_widget_LinearLayout_MEASURED_STATE_MASK
-#define android_widget_LinearLayout_MEASURED_STATE_MASK -16777216L
-#undef android_widget_LinearLayout_MEASURED_HEIGHT_STATE_SHIFT
-#define android_widget_LinearLayout_MEASURED_HEIGHT_STATE_SHIFT 16L
-#undef android_widget_LinearLayout_MEASURED_STATE_TOO_SMALL
-#define android_widget_LinearLayout_MEASURED_STATE_TOO_SMALL 16777216L
-#undef android_widget_LinearLayout_PFLAG2_DRAG_CAN_ACCEPT
-#define android_widget_LinearLayout_PFLAG2_DRAG_CAN_ACCEPT 1L
-#undef android_widget_LinearLayout_PFLAG2_DRAG_HOVERED
-#define android_widget_LinearLayout_PFLAG2_DRAG_HOVERED 2L
-#undef android_widget_LinearLayout_LAYOUT_DIRECTION_LTR
-#define android_widget_LinearLayout_LAYOUT_DIRECTION_LTR 0L
-#undef android_widget_LinearLayout_LAYOUT_DIRECTION_RTL
-#define android_widget_LinearLayout_LAYOUT_DIRECTION_RTL 1L
-#undef android_widget_LinearLayout_LAYOUT_DIRECTION_INHERIT
-#define android_widget_LinearLayout_LAYOUT_DIRECTION_INHERIT 2L
-#undef android_widget_LinearLayout_LAYOUT_DIRECTION_LOCALE
-#define android_widget_LinearLayout_LAYOUT_DIRECTION_LOCALE 3L
-#undef android_widget_LinearLayout_PFLAG2_LAYOUT_DIRECTION_MASK_SHIFT
-#define android_widget_LinearLayout_PFLAG2_LAYOUT_DIRECTION_MASK_SHIFT 2L
-#undef android_widget_LinearLayout_PFLAG2_LAYOUT_DIRECTION_MASK
-#define android_widget_LinearLayout_PFLAG2_LAYOUT_DIRECTION_MASK 12L
-#undef android_widget_LinearLayout_PFLAG2_LAYOUT_DIRECTION_RESOLVED_RTL
-#define android_widget_LinearLayout_PFLAG2_LAYOUT_DIRECTION_RESOLVED_RTL 16L
-#undef android_widget_LinearLayout_PFLAG2_LAYOUT_DIRECTION_RESOLVED
-#define android_widget_LinearLayout_PFLAG2_LAYOUT_DIRECTION_RESOLVED 32L
-#undef android_widget_LinearLayout_PFLAG2_LAYOUT_DIRECTION_RESOLVED_MASK
-#define android_widget_LinearLayout_PFLAG2_LAYOUT_DIRECTION_RESOLVED_MASK 48L
-#undef android_widget_LinearLayout_STATUS_BAR_HIDDEN
-#define android_widget_LinearLayout_STATUS_BAR_HIDDEN 1L
-#undef android_widget_LinearLayout_STATUS_BAR_VISIBLE
-#define android_widget_LinearLayout_STATUS_BAR_VISIBLE 0L
-#undef android_widget_LinearLayout_SYSTEM_UI_FLAG_FULLSCREEN
-#define android_widget_LinearLayout_SYSTEM_UI_FLAG_FULLSCREEN 4L
-#undef android_widget_LinearLayout_SYSTEM_UI_FLAG_HIDE_NAVIGATION
-#define android_widget_LinearLayout_SYSTEM_UI_FLAG_HIDE_NAVIGATION 2L
-#undef android_widget_LinearLayout_SYSTEM_UI_FLAG_IMMERSIVE
-#define android_widget_LinearLayout_SYSTEM_UI_FLAG_IMMERSIVE 2048L
-#undef android_widget_LinearLayout_SYSTEM_UI_FLAG_IMMERSIVE_STICKY
-#define android_widget_LinearLayout_SYSTEM_UI_FLAG_IMMERSIVE_STICKY 4096L
-#undef android_widget_LinearLayout_SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN
-#define android_widget_LinearLayout_SYSTEM_UI_FLAG_LAYOUT_FULLSCREEN 1024L
-#undef android_widget_LinearLayout_SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION
-#define android_widget_LinearLayout_SYSTEM_UI_FLAG_LAYOUT_HIDE_NAVIGATION 512L
-#undef android_widget_LinearLayout_SYSTEM_UI_FLAG_LAYOUT_STABLE
-#define android_widget_LinearLayout_SYSTEM_UI_FLAG_LAYOUT_STABLE 256L
-#undef android_widget_LinearLayout_SYSTEM_UI_FLAG_LOW_PROFILE
-#define android_widget_LinearLayout_SYSTEM_UI_FLAG_LOW_PROFILE 1L
-#undef android_widget_LinearLayout_SYSTEM_UI_FLAG_VISIBLE
-#define android_widget_LinearLayout_SYSTEM_UI_FLAG_VISIBLE 0L
-#undef android_widget_LinearLayout_SYSTEM_UI_LAYOUT_FLAGS
-#define android_widget_LinearLayout_SYSTEM_UI_LAYOUT_FLAGS 1536L
-#undef android_widget_LinearLayout_TEXT_ALIGNMENT_CENTER
-#define android_widget_LinearLayout_TEXT_ALIGNMENT_CENTER 4L
-#undef android_widget_LinearLayout_TEXT_ALIGNMENT_GRAVITY
-#define android_widget_LinearLayout_TEXT_ALIGNMENT_GRAVITY 1L
-#undef android_widget_LinearLayout_TEXT_ALIGNMENT_INHERIT
-#define android_widget_LinearLayout_TEXT_ALIGNMENT_INHERIT 0L
-#undef android_widget_LinearLayout_TEXT_ALIGNMENT_TEXT_END
-#define android_widget_LinearLayout_TEXT_ALIGNMENT_TEXT_END 3L
-#undef android_widget_LinearLayout_TEXT_ALIGNMENT_TEXT_START
-#define android_widget_LinearLayout_TEXT_ALIGNMENT_TEXT_START 2L
-#undef android_widget_LinearLayout_TEXT_ALIGNMENT_VIEW_END
-#define android_widget_LinearLayout_TEXT_ALIGNMENT_VIEW_END 6L
-#undef android_widget_LinearLayout_TEXT_ALIGNMENT_VIEW_START
-#define android_widget_LinearLayout_TEXT_ALIGNMENT_VIEW_START 5L
-#undef android_widget_LinearLayout_TEXT_DIRECTION_ANY_RTL
-#define android_widget_LinearLayout_TEXT_DIRECTION_ANY_RTL 2L
-#undef android_widget_LinearLayout_TEXT_DIRECTION_FIRST_STRONG
-#define android_widget_LinearLayout_TEXT_DIRECTION_FIRST_STRONG 1L
-#undef android_widget_LinearLayout_TEXT_DIRECTION_INHERIT
-#define android_widget_LinearLayout_TEXT_DIRECTION_INHERIT 0L
-#undef android_widget_LinearLayout_TEXT_DIRECTION_LOCALE
-#define android_widget_LinearLayout_TEXT_DIRECTION_LOCALE 5L
-#undef android_widget_LinearLayout_TEXT_DIRECTION_LTR
-#define android_widget_LinearLayout_TEXT_DIRECTION_LTR 3L
-#undef android_widget_LinearLayout_TEXT_DIRECTION_RTL
-#define android_widget_LinearLayout_TEXT_DIRECTION_RTL 4L
-/*
- * Class: android_widget_LinearLayout
- * Method: native_constructor
- * Signature: (Landroid/content/Context;Landroid/util/AttributeSet;)J
- */
-JNIEXPORT jlong JNICALL Java_android_widget_LinearLayout_native_1constructor
- (JNIEnv *, jobject, jobject, jobject);
-
-/*
- * Class: android_widget_LinearLayout
- * Method: setOrientation
- * Signature: (I)V
- */
-JNIEXPORT void JNICALL Java_android_widget_LinearLayout_setOrientation
- (JNIEnv *, jobject, jint);
-
-/*
- * Class: android_widget_LinearLayout
- * Method: getOrientation
- * Signature: ()I
- */
-JNIEXPORT jint JNICALL Java_android_widget_LinearLayout_getOrientation
- (JNIEnv *, jobject);
-
-/*
- * Class: android_widget_LinearLayout
- * Method: native_setHomogenousWeight
- * Signature: (JZ)V
- */
-JNIEXPORT void JNICALL Java_android_widget_LinearLayout_native_1setHomogenousWeight
- (JNIEnv *, jobject, jlong, jboolean);
-
-#ifdef __cplusplus
-}
-#endif
-#endif
diff --git a/src/api-impl-jni/widgets/android_widget_LinearLayout.c b/src/api-impl-jni/widgets/android_widget_LinearLayout.c
deleted file mode 100644
index d8c12279..00000000
--- a/src/api-impl-jni/widgets/android_widget_LinearLayout.c
+++ /dev/null
@@ -1,44 +0,0 @@
-#include
-
-#include "../defines.h"
-#include "../util.h"
-
-#include "WrapperWidget.h"
-
-#include "../generated_headers/android_widget_LinearLayout.h"
-#include "../generated_headers/android_view_ViewGroup.h"
-
-JNIEXPORT jlong JNICALL Java_android_widget_LinearLayout_native_1constructor(JNIEnv *env, jobject this, jobject context, jobject attrs)
-{
- int orientation = attribute_set_get_int(env, attrs, "orientation", NULL, 0);
-
- GtkWidget *wrapper = g_object_ref(wrapper_widget_new());
- GtkWidget *box = gtk_box_new(orientation ? GTK_ORIENTATION_VERTICAL : GTK_ORIENTATION_HORIZONTAL, 0); // spacing of 0
- gtk_box_set_homogeneous(GTK_BOX(box), TRUE);
- wrapper_widget_set_child(WRAPPER_WIDGET(wrapper), box);
- wrapper_widget_set_jobject(WRAPPER_WIDGET(wrapper), env, this);
- gtk_widget_set_name(GTK_WIDGET(box), "LinearLayout");
- if (!attrs) {
- gtk_widget_set_hexpand_set(box, true); // FIXME: to counteract expand on drawing areas
- gtk_widget_set_vexpand_set(box, true); // XXX
- }
- return _INTPTR(box);
-
-// struct ninepatch_t *ninepatch = ninepatch_new("/home/Mis012/Github_and_other_sources/org.happysanta.gd_29_src.tar.gz/res/drawable-mdpi/btn_br_down.9.png");
-// g_object_set_data(G_OBJECT(wrapper), "background_ninepatch", ninepatch);
-}
-
-JNIEXPORT void JNICALL Java_android_widget_LinearLayout_setOrientation(JNIEnv *env, jobject this, jint orientation)
-{
- gtk_orientable_set_orientation(GTK_ORIENTABLE(_PTR(_GET_LONG_FIELD(this, "widget"))), orientation ? GTK_ORIENTATION_VERTICAL : GTK_ORIENTATION_HORIZONTAL);
-}
-
-JNIEXPORT jint JNICALL Java_android_widget_LinearLayout_getOrientation(JNIEnv *env, jobject this)
-{
- return gtk_orientable_get_orientation(GTK_ORIENTABLE(_PTR(_GET_LONG_FIELD(this, "widget"))));
-}
-
-JNIEXPORT void JNICALL Java_android_widget_LinearLayout_native_1setHomogenousWeight(JNIEnv *env, jobject this, jlong widget_ptr, jboolean homogeneous)
-{
- gtk_box_set_homogeneous(_PTR(widget_ptr), homogeneous);
-}
diff --git a/src/api-impl/android/widget/LinearLayout.java b/src/api-impl/android/widget/LinearLayout.java
index 7b32361b..5f32e7a0 100644
--- a/src/api-impl/android/widget/LinearLayout.java
+++ b/src/api-impl/android/widget/LinearLayout.java
@@ -1,58 +1,1805 @@
+/*
+ * Copyright (C) 2006 The Android Open Source Project
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
package android.widget;
import android.content.Context;
+import android.content.res.TypedArray;
+import android.graphics.Canvas;
+import android.graphics.drawable.Drawable;
import android.util.AttributeSet;
+import android.view.Gravity;
import android.view.View;
import android.view.ViewGroup;
+import com.android.internal.R;
+import java.lang.annotation.Retention;
+import java.lang.annotation.RetentionPolicy;
+
+/**
+ * A layout that arranges other views either horizontally in a single column
+ * or vertically in a single row.
+ *
+ * The following snippet shows how to include a linear layout in your layout XML file:
+ *
+ * <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
+ * android:layout_width="match_parent"
+ * android:layout_height="match_parent"
+ * android:paddingLeft="16dp"
+ * android:paddingRight="16dp"
+ * android:orientation="horizontal"
+ * android:gravity="center">
+ *
+ * <!-- Include other widget or layout tags here. These are considered
+ * "child views" or "children" of the linear layout -->
+ *
+ * </LinearLayout>
+ *
+ * Set {@link android.R.styleable#LinearLayout_orientation android:orientation} to specify
+ * whether child views are displayed in a row or column.
+ *
+ * To control how linear layout aligns all the views it contains, set a value for
+ * {@link android.R.styleable#LinearLayout_gravity android:gravity}. For example, the
+ * snippet above sets android:gravity to "center". The value you set affects
+ * both horizontal and vertical alignment of all child views within the single row or column.
+ *
+ * You can set
+ * {@link android.R.styleable#LinearLayout_Layout_layout_weight android:layout_weight}
+ * on individual child views to specify how linear layout divides remaining space amongst
+ * the views it contains. See the
+ * Linear Layout
+ * guide for an example.
+ *
+ * See
+ * {@link android.widget.LinearLayout.LayoutParams LinearLayout.LayoutParams}
+ * to learn about other attributes you can set on a child view to affect its
+ * position and size in the containing linear layout.
+ *
+ * @attr ref android.R.styleable#LinearLayout_baselineAligned
+ * @attr ref android.R.styleable#LinearLayout_baselineAlignedChildIndex
+ * @attr ref android.R.styleable#LinearLayout_gravity
+ * @attr ref android.R.styleable#LinearLayout_measureWithLargestChild
+ * @attr ref android.R.styleable#LinearLayout_orientation
+ * @attr ref android.R.styleable#LinearLayout_weightSum
+ */
public class LinearLayout extends ViewGroup {
+ /** @hide */
+ @Retention(RetentionPolicy.SOURCE)
+ public @interface OrientationMode {}
+ public static final int HORIZONTAL = 0;
+ public static final int VERTICAL = 1;
- public LinearLayout(Context context, AttributeSet attrs) {
- super(context, attrs);
- }
+ /** @hide */
+ @Retention(RetentionPolicy.SOURCE)
+ public @interface DividerMode {}
+
+ /**
+ * Don't show any dividers.
+ */
+ public static final int SHOW_DIVIDER_NONE = 0;
+ /**
+ * Show a divider at the beginning of the group.
+ */
+ public static final int SHOW_DIVIDER_BEGINNING = 1;
+ /**
+ * Show dividers between each item in the group.
+ */
+ public static final int SHOW_DIVIDER_MIDDLE = 2;
+ /**
+ * Show a divider at the end of the group.
+ */
+ public static final int SHOW_DIVIDER_END = 4;
+ /**
+ * Compatibility check. Old versions of the platform would give different
+ * results from measurement passes using EXACTLY and non-EXACTLY modes,
+ * even when the resulting size was the same.
+ */
+ private final boolean mAllowInconsistentMeasurement;
+ /**
+ * Whether the children of this layout are baseline aligned. Only applicable
+ * if {@link #mOrientation} is horizontal.
+ */
+ private boolean mBaselineAligned = true;
+ /**
+ * If this layout is part of another layout that is baseline aligned,
+ * use the child at this index as the baseline.
+ *
+ * Note: this is orthogonal to {@link #mBaselineAligned}, which is concerned
+ * with whether the children of this layout are baseline aligned.
+ */
+ private int mBaselineAlignedChildIndex = -1;
+ /**
+ * The additional offset to the child's baseline.
+ * We'll calculate the baseline of this layout as we measure vertically; for
+ * horizontal linear layouts, the offset of 0 is appropriate.
+ */
+ private int mBaselineChildTop = 0;
+ private int mOrientation;
+
+ private int mGravity = Gravity.START | Gravity.TOP;
+ private int mTotalLength;
+ private float mWeightSum;
+ private boolean mUseLargestChild;
+ private int[] mMaxAscent;
+ private int[] mMaxDescent;
+ private static final int VERTICAL_GRAVITY_COUNT = 4;
+ private static final int INDEX_CENTER_VERTICAL = 0;
+ private static final int INDEX_TOP = 1;
+ private static final int INDEX_BOTTOM = 2;
+ private static final int INDEX_FILL = 3;
+ private Drawable mDivider;
+ private int mDividerWidth;
+ private int mDividerHeight;
+ private int mShowDividers;
+ private int mDividerPadding;
+ /**
+ * Signals that compatibility booleans have been initialized according to
+ * target SDK versions.
+ */
+ private static boolean sCompatibilityDone = false;
+ /**
+ * Behavior change in P; always remeasure weighted children, regardless of excess space.
+ */
+ private static boolean sRemeasureWeightedChildren = true;
public LinearLayout(Context context) {
- super(context);
+ this(context, null);
+ }
+
+ public LinearLayout(Context context, AttributeSet attrs) {
+ this(context, attrs, 0);
+ }
+
+ public LinearLayout(Context context, AttributeSet attrs, int defStyleAttr) {
+ this(context, attrs, defStyleAttr, 0);
+ }
+
+ public LinearLayout(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
+ super(context, attrs, defStyleAttr, defStyleRes);
+ if (!sCompatibilityDone && context != null) {
+ final int targetSdkVersion = context.getApplicationInfo().targetSdkVersion;
+ // Older apps only remeasure non-zero children
+ sRemeasureWeightedChildren = targetSdkVersion >= /*Build.VERSION_CODES.P*/ 28;
+ sCompatibilityDone = true;
+ }
+ final TypedArray a = context.obtainStyledAttributes(
+ attrs, com.android.internal.R.styleable.LinearLayout, defStyleAttr, defStyleRes);
+ // saveAttributeDataForStyleable(context, com.android.internal.R.styleable.LinearLayout,
+ // attrs, a, defStyleAttr, defStyleRes);
+ int index = a.getInt(com.android.internal.R.styleable.LinearLayout_orientation, -1);
+ if (index >= 0) {
+ setOrientation(index);
+ }
+ index = a.getInt(com.android.internal.R.styleable.LinearLayout_gravity, -1);
+ if (index >= 0) {
+ setGravity(index);
+ }
+ boolean baselineAligned = a.getBoolean(R.styleable.LinearLayout_baselineAligned, true);
+ if (!baselineAligned) {
+ setBaselineAligned(baselineAligned);
+ }
+ mWeightSum = a.getFloat(R.styleable.LinearLayout_weightSum, -1.0f);
+ mBaselineAlignedChildIndex =
+ a.getInt(com.android.internal.R.styleable.LinearLayout_baselineAlignedChildIndex, -1);
+ mUseLargestChild = a.getBoolean(R.styleable.LinearLayout_measureWithLargestChild, false);
+ mShowDividers = a.getInt(R.styleable.LinearLayout_showDividers, SHOW_DIVIDER_NONE);
+ mDividerPadding = a.getDimensionPixelSize(R.styleable.LinearLayout_dividerPadding, 0);
+ setDividerDrawable(a.getDrawable(R.styleable.LinearLayout_divider));
+ final int version = context.getApplicationInfo().targetSdkVersion;
+ mAllowInconsistentMeasurement = version <= /*Build.VERSION_CODES.M*/ 23;
+ a.recycle();
+ }
+
+ /**
+ * Returns true if this layout is currently configured to show at least one
+ * divider.
+ */
+ private boolean isShowingDividers() {
+ return (mShowDividers != SHOW_DIVIDER_NONE) && (mDivider != null);
+ }
+
+ /**
+ * Set how dividers should be shown between items in this layout
+ *
+ * @param showDividers One or more of {@link #SHOW_DIVIDER_BEGINNING},
+ * {@link #SHOW_DIVIDER_MIDDLE}, or {@link #SHOW_DIVIDER_END}
+ * to show dividers, or {@link #SHOW_DIVIDER_NONE} to show no dividers.
+ */
+ public void setShowDividers(@DividerMode int showDividers) {
+ if (showDividers == mShowDividers) {
+ return;
+ }
+ mShowDividers = showDividers;
+ setWillNotDraw(!isShowingDividers());
+ requestLayout();
+ }
+
+ public boolean shouldDelayChildPressedState() {
+ return false;
+ }
+
+ /**
+ * @return A flag set indicating how dividers should be shown around items.
+ * @see #setShowDividers(int)
+ */
+ @DividerMode
+ public int getShowDividers() {
+ return mShowDividers;
+ }
+
+ /**
+ * @return the divider Drawable that will divide each item.
+ *
+ * @see #setDividerDrawable(Drawable)
+ *
+ * @attr ref android.R.styleable#LinearLayout_divider
+ */
+ public Drawable getDividerDrawable() {
+ return mDivider;
+ }
+
+ /**
+ * Set a drawable to be used as a divider between items.
+ *
+ * @param divider Drawable that will divide each item.
+ *
+ * @see #setShowDividers(int)
+ *
+ * @attr ref android.R.styleable#LinearLayout_divider
+ */
+ public void setDividerDrawable(Drawable divider) {
+ if (divider == mDivider) {
+ return;
+ }
+ mDivider = divider;
+ if (divider != null) {
+ mDividerWidth = divider.getIntrinsicWidth();
+ mDividerHeight = divider.getIntrinsicHeight();
+ } else {
+ mDividerWidth = 0;
+ mDividerHeight = 0;
+ }
+ setWillNotDraw(!isShowingDividers());
+ requestLayout();
+ }
+
+ /**
+ * Set padding displayed on both ends of dividers. For a vertical layout, the padding is applied
+ * to left and right end of dividers. For a horizontal layout, the padding is applied to top and
+ * bottom end of dividers.
+ *
+ * @param padding Padding value in pixels that will be applied to each end
+ *
+ * @see #setShowDividers(int)
+ * @see #setDividerDrawable(Drawable)
+ * @see #getDividerPadding()
+ */
+ public void setDividerPadding(int padding) {
+ if (padding == mDividerPadding) {
+ return;
+ }
+ mDividerPadding = padding;
+ if (isShowingDividers()) {
+ requestLayout();
+ invalidate();
+ }
+ }
+
+ /**
+ * Get the padding size used to inset dividers in pixels
+ *
+ * @see #setShowDividers(int)
+ * @see #setDividerDrawable(Drawable)
+ * @see #setDividerPadding(int)
+ */
+ public int getDividerPadding() {
+ return mDividerPadding;
+ }
+ /**
+ * Get the width of the current divider drawable.
+ *
+ * @hide Used internally by framework.
+ */
+ public int getDividerWidth() {
+ return mDividerWidth;
+ }
+
+ /* disabled for now to not create lots of SkiaAreas */
+ // @Override
+ // protected void onDraw(Canvas canvas) {
+ // if (mDivider == null) {
+ // return;
+ // }
+ // if (mOrientation == VERTICAL) {
+ // drawDividersVertical(canvas);
+ // } else {
+ // drawDividersHorizontal(canvas);
+ // }
+ // }
+
+ void drawDividersVertical(Canvas canvas) {
+ final int count = getVirtualChildCount();
+ for (int i = 0; i < count; i++) {
+ final View child = getVirtualChildAt(i);
+ if (child != null && child.getVisibility() != GONE) {
+ if (hasDividerBeforeChildAt(i)) {
+ final LayoutParams lp = (LayoutParams) child.getLayoutParams();
+ final int top = child.getTop() - lp.topMargin - mDividerHeight;
+ drawHorizontalDivider(canvas, top);
+ }
+ }
+ }
+ if (hasDividerBeforeChildAt(count)) {
+ final View child = getLastNonGoneChild();
+ int bottom = 0;
+ if (child == null) {
+ bottom = getHeight() - getPaddingBottom() - mDividerHeight;
+ } else {
+ final LayoutParams lp = (LayoutParams) child.getLayoutParams();
+ bottom = child.getBottom() + lp.bottomMargin;
+ }
+ drawHorizontalDivider(canvas, bottom);
+ }
+ }
+
+ /**
+ * Finds the last child that is not gone. The last child will be used as the reference for
+ * where the end divider should be drawn.
+ */
+ private View getLastNonGoneChild() {
+ for (int i = getVirtualChildCount() - 1; i >= 0; i--) {
+ final View child = getVirtualChildAt(i);
+ if (child != null && child.getVisibility() != GONE) {
+ return child;
+ }
+ }
+ return null;
+ }
+
+ void drawDividersHorizontal(Canvas canvas) {
+ final int count = getVirtualChildCount();
+ final boolean isLayoutRtl = /*isLayoutRtl()*/ false;
+ for (int i = 0; i < count; i++) {
+ final View child = getVirtualChildAt(i);
+ if (child != null && child.getVisibility() != GONE) {
+ if (hasDividerBeforeChildAt(i)) {
+ final LayoutParams lp = (LayoutParams) child.getLayoutParams();
+ final int position;
+ if (isLayoutRtl) {
+ position = child.getRight() + lp.rightMargin;
+ } else {
+ position = child.getLeft() - lp.leftMargin - mDividerWidth;
+ }
+ drawVerticalDivider(canvas, position);
+ }
+ }
+ }
+ if (hasDividerBeforeChildAt(count)) {
+ final View child = getLastNonGoneChild();
+ int position;
+ if (child == null) {
+ if (isLayoutRtl) {
+ position = getPaddingLeft();
+ } else {
+ position = getWidth() - getPaddingRight() - mDividerWidth;
+ }
+ } else {
+ final LayoutParams lp = (LayoutParams) child.getLayoutParams();
+ if (isLayoutRtl) {
+ position = child.getLeft() - lp.leftMargin - mDividerWidth;
+ } else {
+ position = child.getRight() + lp.rightMargin;
+ }
+ }
+ drawVerticalDivider(canvas, position);
+ }
+ }
+
+ void drawHorizontalDivider(Canvas canvas, int top) {
+ mDivider.setBounds(getPaddingLeft() + mDividerPadding, top,
+ getWidth() - getPaddingRight() - mDividerPadding, top + mDividerHeight);
+ mDivider.draw(canvas);
+ }
+
+ void drawVerticalDivider(Canvas canvas, int left) {
+ mDivider.setBounds(left, getPaddingTop() + mDividerPadding,
+ left + mDividerWidth, getHeight() - getPaddingBottom() - mDividerPadding);
+ mDivider.draw(canvas);
+ }
+
+ /**
+ * Indicates whether widgets contained within this layout are aligned
+ * on their baseline or not.
+ *
+ * @return true when widgets are baseline-aligned, false otherwise
+ */
+ public boolean isBaselineAligned() {
+ return mBaselineAligned;
+ }
+
+ /**
+ * Defines whether widgets contained in this layout are
+ * baseline-aligned or not.
+ *
+ * @param baselineAligned true to align widgets on their baseline,
+ * false otherwise
+ *
+ * @attr ref android.R.styleable#LinearLayout_baselineAligned
+ */
+ public void setBaselineAligned(boolean baselineAligned) {
+ mBaselineAligned = baselineAligned;
+ }
+
+ /**
+ * When true, all children with a weight will be considered having
+ * the minimum size of the largest child. If false, all children are
+ * measured normally.
+ *
+ * @return True to measure children with a weight using the minimum
+ * size of the largest child, false otherwise.
+ *
+ * @attr ref android.R.styleable#LinearLayout_measureWithLargestChild
+ */
+ public boolean isMeasureWithLargestChildEnabled() {
+ return mUseLargestChild;
+ }
+
+ /**
+ * When set to true, all children with a weight will be considered having
+ * the minimum size of the largest child. If false, all children are
+ * measured normally.
+ *
+ * Disabled by default.
+ *
+ * @param enabled True to measure children with a weight using the
+ * minimum size of the largest child, false otherwise.
+ *
+ * @attr ref android.R.styleable#LinearLayout_measureWithLargestChild
+ */
+ public void setMeasureWithLargestChildEnabled(boolean enabled) {
+ mUseLargestChild = enabled;
}
@Override
- protected native long native_constructor(Context context, AttributeSet attrs);
-
- public native void setOrientation(int orientation);
- public native int getOrientation();
- protected native void native_setHomogenousWeight(long widget, boolean homogenous);
- public void setWeightSum(float weightSum) {}
-
- @Override
- protected void addViewInternal(View child, int index, ViewGroup.LayoutParams params) {
- super.addViewInternal(child, index, params);
- // check if all children have the same weight and set GtkBox to homogeneous if so
- float weight = params.weight;
- for (int i = 0; i < children.size(); i++) {
- if (children.get(i).getLayoutParams().weight != weight) {
- weight = 0;
- break;
+ public int getBaseline() {
+ if (mBaselineAlignedChildIndex < 0) {
+ return super.getBaseline();
+ }
+ if (getChildCount() <= mBaselineAlignedChildIndex) {
+ throw new RuntimeException("mBaselineAlignedChildIndex of LinearLayout "
+ + "set to an index that is out of bounds.");
+ }
+ final View child = getChildAt(mBaselineAlignedChildIndex);
+ final int childBaseline = child.getBaseline();
+ if (childBaseline == -1) {
+ if (mBaselineAlignedChildIndex == 0) {
+ // this is just the default case, safe to return -1
+ return -1;
+ }
+ // the user picked an index that points to something that doesn't
+ // know how to calculate its baseline.
+ throw new RuntimeException("mBaselineAlignedChildIndex of LinearLayout "
+ + "points to a View that doesn't know how to get its baseline.");
+ }
+ // TODO: This should try to take into account the virtual offsets
+ // (See getNextLocationOffset and getLocationOffset)
+ // We should add to childTop:
+ // sum([getNextLocationOffset(getChildAt(i)) / i < mBaselineAlignedChildIndex])
+ // and also add:
+ // getLocationOffset(child)
+ int childTop = mBaselineChildTop;
+ if (mOrientation == VERTICAL) {
+ final int majorGravity = mGravity & Gravity.VERTICAL_GRAVITY_MASK;
+ if (majorGravity != Gravity.TOP) {
+ switch (majorGravity) {
+ case Gravity.BOTTOM:
+ childTop = getBottom() - getTop() - /*mPaddingBottom*/ 0 - mTotalLength;
+ break;
+ case Gravity.CENTER_VERTICAL:
+ childTop += ((getBottom() - getTop() - /*mPaddingTop*/ 0 - /*mPaddingBottom*/ 0) -
+ mTotalLength) / 2;
+ break;
+ }
}
}
- native_setHomogenousWeight(widget, weight > 0);
+ LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams) child.getLayoutParams();
+ return childTop + lp.topMargin + childBaseline;
+ }
+
+ /**
+ * @return The index of the child that will be used if this layout is
+ * part of a larger layout that is baseline aligned, or -1 if none has
+ * been set.
+ */
+ public int getBaselineAlignedChildIndex() {
+ return mBaselineAlignedChildIndex;
+ }
+
+ /**
+ * @param i The index of the child that will be used if this layout is
+ * part of a larger layout that is baseline aligned.
+ *
+ * @attr ref android.R.styleable#LinearLayout_baselineAlignedChildIndex
+ */
+ public void setBaselineAlignedChildIndex(int i) {
+ if ((i < 0) || (i >= getChildCount())) {
+ throw new IllegalArgumentException("base aligned child index out "
+ + "of range (0, " + getChildCount() + ")");
+ }
+ mBaselineAlignedChildIndex = i;
+ }
+
+ /**
+ * Returns the view at the specified index. This method can be overridden
+ * to take into account virtual children. Refer to
+ * {@link android.widget.TableLayout} and {@link android.widget.TableRow}
+ * for an example.
+ *
+ * @param index the child's index
+ * @return the child at the specified index, may be {@code null}
+ */
+ View getVirtualChildAt(int index) {
+ return getChildAt(index);
+ }
+
+ /**
+ * Returns the virtual number of children. This number might be different
+ * than the actual number of children if the layout can hold virtual
+ * children. Refer to
+ * {@link android.widget.TableLayout} and {@link android.widget.TableRow}
+ * for an example.
+ *
+ * @return the virtual number of children
+ */
+ int getVirtualChildCount() {
+ return getChildCount();
+ }
+
+ /**
+ * Returns the desired weights sum.
+ *
+ * @return A number greater than 0.0f if the weight sum is defined, or
+ * a number lower than or equals to 0.0f if not weight sum is
+ * to be used.
+ */
+ public float getWeightSum() {
+ return mWeightSum;
+ }
+
+ /**
+ * Defines the desired weights sum. If unspecified the weights sum is computed
+ * at layout time by adding the layout_weight of each child.
+ *
+ * This can be used for instance to give a single child 50% of the total
+ * available space by giving it a layout_weight of 0.5 and setting the
+ * weightSum to 1.0.
+ *
+ * @param weightSum a number greater than 0.0f, or a number lower than or equals
+ * to 0.0f if the weight sum should be computed from the children's
+ * layout_weight
+ */
+ public void setWeightSum(float weightSum) {
+ mWeightSum = Math.max(0.0f, weightSum);
+ }
+
+ @Override
+ protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
+ if (mOrientation == VERTICAL) {
+ measureVertical(widthMeasureSpec, heightMeasureSpec);
+ } else {
+ measureHorizontal(widthMeasureSpec, heightMeasureSpec);
+ }
+ }
+
+ /**
+ * Determines where to position dividers between children.
+ *
+ * @param childIndex Index of child to check for preceding divider
+ * @return true if there should be a divider before the child at childIndex
+ * @hide Pending API consideration. Currently only used internally by the system.
+ */
+ protected boolean hasDividerBeforeChildAt(int childIndex) {
+ if (mShowDividers == SHOW_DIVIDER_NONE) {
+ // Short-circuit to save iteration over child views.
+ return false;
+ }
+ if (childIndex == getVirtualChildCount()) {
+ // Check whether the end divider should draw.
+ return (mShowDividers & SHOW_DIVIDER_END) != 0;
+ }
+ boolean allViewsAreGoneBefore = allViewsAreGoneBefore(childIndex);
+ if (allViewsAreGoneBefore) {
+ // This is the first view that's not gone, check if beginning divider is enabled.
+ return (mShowDividers & SHOW_DIVIDER_BEGINNING) != 0;
+ } else {
+ return (mShowDividers & SHOW_DIVIDER_MIDDLE) != 0;
+ }
+ }
+
+ /**
+ * Determines whether or not there's a divider after a specified child index.
+ *
+ * @param childIndex Index of child to check for following divider
+ * @return true if there should be a divider after the child at childIndex
+ */
+ private boolean hasDividerAfterChildAt(int childIndex) {
+ if (mShowDividers == SHOW_DIVIDER_NONE) {
+ // Short-circuit to save iteration over child views.
+ return false;
+ }
+ if (allViewsAreGoneAfter(childIndex)) {
+ // This is the last view that's not gone, check if end divider is enabled.
+ return (mShowDividers & SHOW_DIVIDER_END) != 0;
+ }
+ return (mShowDividers & SHOW_DIVIDER_MIDDLE) != 0;
+ }
+
+ /**
+ * Checks whether all (virtual) child views before the given index are gone.
+ */
+ private boolean allViewsAreGoneBefore(int childIndex) {
+ for (int i = childIndex - 1; i >= 0; i--) {
+ final View child = getVirtualChildAt(i);
+ if (child != null && child.getVisibility() != GONE) {
+ return false;
+ }
+ }
+ return true;
+ }
+
+ /**
+ * Checks whether all (virtual) child views after the given index are gone.
+ */
+ private boolean allViewsAreGoneAfter(int childIndex) {
+ final int count = getVirtualChildCount();
+ for (int i = childIndex + 1; i < count; i++) {
+ final View child = getVirtualChildAt(i);
+ if (child != null && child.getVisibility() != GONE) {
+ return false;
+ }
+ }
+ return true;
+ }
+
+ /**
+ * Measures the children when the orientation of this LinearLayout is set
+ * to {@link #VERTICAL}.
+ *
+ * @param widthMeasureSpec Horizontal space requirements as imposed by the parent.
+ * @param heightMeasureSpec Vertical space requirements as imposed by the parent.
+ *
+ * @see #getOrientation()
+ * @see #setOrientation(int)
+ * @see #onMeasure(int, int)
+ */
+ void measureVertical(int widthMeasureSpec, int heightMeasureSpec) {
+ mTotalLength = 0;
+ int maxWidth = 0;
+ int childState = 0;
+ int alternativeMaxWidth = 0;
+ int weightedMaxWidth = 0;
+ boolean allFillParent = true;
+ float totalWeight = 0;
+ final int count = getVirtualChildCount();
+ final int widthMode = MeasureSpec.getMode(widthMeasureSpec);
+ final int heightMode = MeasureSpec.getMode(heightMeasureSpec);
+ boolean matchWidth = false;
+ boolean skippedMeasure = false;
+ final int baselineChildIndex = mBaselineAlignedChildIndex;
+ final boolean useLargestChild = mUseLargestChild;
+ int largestChildHeight = Integer.MIN_VALUE;
+ int consumedExcessSpace = 0;
+ int nonSkippedChildCount = 0;
+ // See how tall everyone is. Also remember max width.
+ for (int i = 0; i < count; ++i) {
+ final View child = getVirtualChildAt(i);
+ if (child == null) {
+ mTotalLength += measureNullChild(i);
+ continue;
+ }
+ if (child.getVisibility() == View.GONE) {
+ i += getChildrenSkipCount(child, i);
+ continue;
+ }
+ nonSkippedChildCount++;
+ if (hasDividerBeforeChildAt(i)) {
+ mTotalLength += mDividerHeight;
+ }
+ final LayoutParams lp = (LayoutParams) child.getLayoutParams();
+ totalWeight += lp.weight;
+ final boolean useExcessSpace = lp.height == 0 && lp.weight > 0;
+ if (heightMode == MeasureSpec.EXACTLY && useExcessSpace) {
+ // Optimization: don't bother measuring children who are only
+ // laid out using excess space. These views will get measured
+ // later if we have space to distribute.
+ final int totalLength = mTotalLength;
+ mTotalLength = Math.max(totalLength, totalLength + lp.topMargin + lp.bottomMargin);
+ skippedMeasure = true;
+ } else {
+ if (useExcessSpace) {
+ // The heightMode is either UNSPECIFIED or AT_MOST, and
+ // this child is only laid out using excess space. Measure
+ // using WRAP_CONTENT so that we can find out the view's
+ // optimal height. We'll restore the original height of 0
+ // after measurement.
+ lp.height = LayoutParams.WRAP_CONTENT;
+ }
+ // Determine how big this child would like to be. If this or
+ // previous children have given a weight, then we allow it to
+ // use all available space (and we will shrink things later
+ // if needed).
+ final int usedHeight = totalWeight == 0 ? mTotalLength : 0;
+ measureChildBeforeLayout(child, i, widthMeasureSpec, 0,
+ heightMeasureSpec, usedHeight);
+ final int childHeight = child.getMeasuredHeight();
+ if (useExcessSpace) {
+ // Restore the original height and record how much space
+ // we've allocated to excess-only children so that we can
+ // match the behavior of EXACTLY measurement.
+ lp.height = 0;
+ consumedExcessSpace += childHeight;
+ }
+ final int totalLength = mTotalLength;
+ mTotalLength = Math.max(totalLength, totalLength + childHeight + lp.topMargin +
+ lp.bottomMargin + getNextLocationOffset(child));
+ if (useLargestChild) {
+ largestChildHeight = Math.max(childHeight, largestChildHeight);
+ }
+ }
+ /**
+ * If applicable, compute the additional offset to the child's baseline
+ * we'll need later when asked {@link #getBaseline}.
+ */
+ if ((baselineChildIndex >= 0) && (baselineChildIndex == i + 1)) {
+ mBaselineChildTop = mTotalLength;
+ }
+ // if we are trying to use a child index for our baseline, the above
+ // book keeping only works if there are no children above it with
+ // weight. fail fast to aid the developer.
+ if (i < baselineChildIndex && lp.weight > 0) {
+ throw new RuntimeException("A child of LinearLayout with index "
+ + "less than mBaselineAlignedChildIndex has weight > 0, which "
+ + "won't work. Either remove the weight, or don't set "
+ + "mBaselineAlignedChildIndex.");
+ }
+ boolean matchWidthLocally = false;
+ if (widthMode != MeasureSpec.EXACTLY && lp.width == LayoutParams.MATCH_PARENT) {
+ // The width of the linear layout will scale, and at least one
+ // child said it wanted to match our width. Set a flag
+ // indicating that we need to remeasure at least that view when
+ // we know our width.
+ matchWidth = true;
+ matchWidthLocally = true;
+ }
+ final int margin = lp.leftMargin + lp.rightMargin;
+ final int measuredWidth = child.getMeasuredWidth() + margin;
+ maxWidth = Math.max(maxWidth, measuredWidth);
+ childState = combineMeasuredStates(childState, child.getMeasuredState());
+ allFillParent = allFillParent && lp.width == LayoutParams.MATCH_PARENT;
+ if (lp.weight > 0) {
+ /*
+ * Widths of weighted Views are bogus if we end up
+ * remeasuring, so keep them separate.
+ */
+ weightedMaxWidth = Math.max(weightedMaxWidth,
+ matchWidthLocally ? margin : measuredWidth);
+ } else {
+ alternativeMaxWidth = Math.max(alternativeMaxWidth,
+ matchWidthLocally ? margin : measuredWidth);
+ }
+ i += getChildrenSkipCount(child, i);
+ }
+ if (nonSkippedChildCount > 0 && hasDividerBeforeChildAt(count)) {
+ mTotalLength += mDividerHeight;
+ }
+ if (useLargestChild &&
+ (heightMode == MeasureSpec.AT_MOST || heightMode == MeasureSpec.UNSPECIFIED)) {
+ mTotalLength = 0;
+ for (int i = 0; i < count; ++i) {
+ final View child = getVirtualChildAt(i);
+ if (child == null) {
+ mTotalLength += measureNullChild(i);
+ continue;
+ }
+ if (child.getVisibility() == GONE) {
+ i += getChildrenSkipCount(child, i);
+ continue;
+ }
+ final LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams)
+ child.getLayoutParams();
+ // Account for negative margins
+ final int totalLength = mTotalLength;
+ mTotalLength = Math.max(totalLength, totalLength + largestChildHeight +
+ lp.topMargin + lp.bottomMargin + getNextLocationOffset(child));
+ }
+ }
+ // Add in our padding
+ mTotalLength += /*mPaddingTop*/ 0 + /*mPaddingBottom*/ 0;
+ int heightSize = mTotalLength;
+ // Check against our minimum height
+ heightSize = Math.max(heightSize, getSuggestedMinimumHeight());
+ // Reconcile our calculated size with the heightMeasureSpec
+ int heightSizeAndState = resolveSizeAndState(heightSize, heightMeasureSpec, 0);
+ heightSize = heightSizeAndState & MEASURED_SIZE_MASK;
+ // Either expand children with weight to take up available space or
+ // shrink them if they extend beyond our current bounds. If we skipped
+ // measurement on any children, we need to measure them now.
+ int remainingExcess = heightSize - mTotalLength
+ + (mAllowInconsistentMeasurement ? 0 : consumedExcessSpace);
+ if (skippedMeasure
+ || ((sRemeasureWeightedChildren || remainingExcess != 0) && totalWeight > 0.0f)) {
+ float remainingWeightSum = mWeightSum > 0.0f ? mWeightSum : totalWeight;
+ mTotalLength = 0;
+ for (int i = 0; i < count; ++i) {
+ final View child = getVirtualChildAt(i);
+ if (child == null || child.getVisibility() == View.GONE) {
+ continue;
+ }
+ final LayoutParams lp = (LayoutParams) child.getLayoutParams();
+ final float childWeight = lp.weight;
+ if (childWeight > 0) {
+ final int share = (int) (childWeight * remainingExcess / remainingWeightSum);
+ remainingExcess -= share;
+ remainingWeightSum -= childWeight;
+ final int childHeight;
+ if (mUseLargestChild && heightMode != MeasureSpec.EXACTLY) {
+ childHeight = largestChildHeight;
+ } else if (lp.height == 0 && (!mAllowInconsistentMeasurement
+ || heightMode == MeasureSpec.EXACTLY)) {
+ // This child needs to be laid out from scratch using
+ // only its share of excess space.
+ childHeight = share;
+ } else {
+ // This child had some intrinsic height to which we
+ // need to add its share of excess space.
+ childHeight = child.getMeasuredHeight() + share;
+ }
+ final int childHeightMeasureSpec = MeasureSpec.makeMeasureSpec(
+ Math.max(0, childHeight), MeasureSpec.EXACTLY);
+ final int childWidthMeasureSpec = getChildMeasureSpec(widthMeasureSpec,
+ /*mPaddingLeft*/ 0 + /*mPaddingRight*/ 0 + lp.leftMargin + lp.rightMargin,
+ lp.width);
+ child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
+ // Child may now not fit in vertical dimension.
+ childState = combineMeasuredStates(childState, child.getMeasuredState()
+ & (MEASURED_STATE_MASK>>MEASURED_HEIGHT_STATE_SHIFT));
+ }
+ final int margin = lp.leftMargin + lp.rightMargin;
+ final int measuredWidth = child.getMeasuredWidth() + margin;
+ maxWidth = Math.max(maxWidth, measuredWidth);
+ boolean matchWidthLocally = widthMode != MeasureSpec.EXACTLY &&
+ lp.width == LayoutParams.MATCH_PARENT;
+ alternativeMaxWidth = Math.max(alternativeMaxWidth,
+ matchWidthLocally ? margin : measuredWidth);
+ allFillParent = allFillParent && lp.width == LayoutParams.MATCH_PARENT;
+ final int totalLength = mTotalLength;
+ mTotalLength = Math.max(totalLength, totalLength + child.getMeasuredHeight() +
+ lp.topMargin + lp.bottomMargin + getNextLocationOffset(child));
+ }
+ // Add in our padding
+ mTotalLength += /*mPaddingTop*/ 0 + /*mPaddingBottom*/ 0;
+ // TODO: Should we recompute the heightSpec based on the new total length?
+ } else {
+ alternativeMaxWidth = Math.max(alternativeMaxWidth,
+ weightedMaxWidth);
+ // We have no limit, so make all weighted views as tall as the largest child.
+ // Children will have already been measured once.
+ if (useLargestChild && heightMode != MeasureSpec.EXACTLY) {
+ for (int i = 0; i < count; i++) {
+ final View child = getVirtualChildAt(i);
+ if (child == null || child.getVisibility() == View.GONE) {
+ continue;
+ }
+ final LinearLayout.LayoutParams lp =
+ (LinearLayout.LayoutParams) child.getLayoutParams();
+ float childExtra = lp.weight;
+ if (childExtra > 0) {
+ child.measure(
+ MeasureSpec.makeMeasureSpec(child.getMeasuredWidth(),
+ MeasureSpec.EXACTLY),
+ MeasureSpec.makeMeasureSpec(largestChildHeight,
+ MeasureSpec.EXACTLY));
+ }
+ }
+ }
+ }
+ if (!allFillParent && widthMode != MeasureSpec.EXACTLY) {
+ maxWidth = alternativeMaxWidth;
+ }
+ maxWidth += /*mPaddingLeft*/ 0 + /*mPaddingRight*/ 0;
+ // Check against our minimum width
+ maxWidth = Math.max(maxWidth, getSuggestedMinimumWidth());
+ setMeasuredDimension(resolveSizeAndState(maxWidth, widthMeasureSpec, childState),
+ heightSizeAndState);
+ if (matchWidth) {
+ forceUniformWidth(count, heightMeasureSpec);
+ }
+ }
+
+ private void forceUniformWidth(int count, int heightMeasureSpec) {
+ // Pretend that the linear layout has an exact size.
+ int uniformMeasureSpec = MeasureSpec.makeMeasureSpec(getMeasuredWidth(),
+ MeasureSpec.EXACTLY);
+ for (int i = 0; i< count; ++i) {
+ final View child = getVirtualChildAt(i);
+ if (child != null && child.getVisibility() != GONE) {
+ LinearLayout.LayoutParams lp = ((LinearLayout.LayoutParams)child.getLayoutParams());
+ if (lp.width == LayoutParams.MATCH_PARENT) {
+ // Temporarily force children to reuse their old measured height
+ // FIXME: this may not be right for something like wrapping text?
+ int oldHeight = lp.height;
+ lp.height = child.getMeasuredHeight();
+ // Remeasue with new dimensions
+ measureChildWithMargins(child, uniformMeasureSpec, 0, heightMeasureSpec, 0);
+ lp.height = oldHeight;
+ }
+ }
+ }
+ }
+
+ /**
+ * Measures the children when the orientation of this LinearLayout is set
+ * to {@link #HORIZONTAL}.
+ *
+ * @param widthMeasureSpec Horizontal space requirements as imposed by the parent.
+ * @param heightMeasureSpec Vertical space requirements as imposed by the parent.
+ *
+ * @see #getOrientation()
+ * @see #setOrientation(int)
+ * @see #onMeasure(int, int)
+ */
+ void measureHorizontal(int widthMeasureSpec, int heightMeasureSpec) {
+ mTotalLength = 0;
+ int maxHeight = 0;
+ int childState = 0;
+ int alternativeMaxHeight = 0;
+ int weightedMaxHeight = 0;
+ boolean allFillParent = true;
+ float totalWeight = 0;
+ final int count = getVirtualChildCount();
+ final int widthMode = MeasureSpec.getMode(widthMeasureSpec);
+ final int heightMode = MeasureSpec.getMode(heightMeasureSpec);
+ boolean matchHeight = false;
+ boolean skippedMeasure = false;
+ if (mMaxAscent == null || mMaxDescent == null) {
+ mMaxAscent = new int[VERTICAL_GRAVITY_COUNT];
+ mMaxDescent = new int[VERTICAL_GRAVITY_COUNT];
+ }
+ final int[] maxAscent = mMaxAscent;
+ final int[] maxDescent = mMaxDescent;
+ maxAscent[0] = maxAscent[1] = maxAscent[2] = maxAscent[3] = -1;
+ maxDescent[0] = maxDescent[1] = maxDescent[2] = maxDescent[3] = -1;
+ final boolean baselineAligned = mBaselineAligned;
+ final boolean useLargestChild = mUseLargestChild;
+ final boolean isExactly = widthMode == MeasureSpec.EXACTLY;
+ int largestChildWidth = Integer.MIN_VALUE;
+ int usedExcessSpace = 0;
+ int nonSkippedChildCount = 0;
+ // See how wide everyone is. Also remember max height.
+ for (int i = 0; i < count; ++i) {
+ final View child = getVirtualChildAt(i);
+ if (child == null) {
+ mTotalLength += measureNullChild(i);
+ continue;
+ }
+ if (child.getVisibility() == GONE) {
+ i += getChildrenSkipCount(child, i);
+ continue;
+ }
+ nonSkippedChildCount++;
+ if (hasDividerBeforeChildAt(i)) {
+ mTotalLength += mDividerWidth;
+ }
+ final LayoutParams lp = (LayoutParams) child.getLayoutParams();
+ totalWeight += lp.weight;
+ final boolean useExcessSpace = lp.width == 0 && lp.weight > 0;
+ if (widthMode == MeasureSpec.EXACTLY && useExcessSpace) {
+ // Optimization: don't bother measuring children who are only
+ // laid out using excess space. These views will get measured
+ // later if we have space to distribute.
+ if (isExactly) {
+ mTotalLength += lp.leftMargin + lp.rightMargin;
+ } else {
+ final int totalLength = mTotalLength;
+ mTotalLength = Math.max(totalLength, totalLength +
+ lp.leftMargin + lp.rightMargin);
+ }
+ // Baseline alignment requires to measure widgets to obtain the
+ // baseline offset (in particular for TextViews). The following
+ // defeats the optimization mentioned above. Allow the child to
+ // use as much space as it wants because we can shrink things
+ // later (and re-measure).
+ if (baselineAligned) {
+ final int freeWidthSpec = MeasureSpec.makeMeasureSpec(
+ MeasureSpec.getSize(widthMeasureSpec), MeasureSpec.UNSPECIFIED);
+ final int freeHeightSpec = MeasureSpec.makeMeasureSpec(
+ MeasureSpec.getSize(heightMeasureSpec), MeasureSpec.UNSPECIFIED);
+ child.measure(freeWidthSpec, freeHeightSpec);
+ } else {
+ skippedMeasure = true;
+ }
+ } else {
+ if (useExcessSpace) {
+ // The widthMode is either UNSPECIFIED or AT_MOST, and
+ // this child is only laid out using excess space. Measure
+ // using WRAP_CONTENT so that we can find out the view's
+ // optimal width. We'll restore the original width of 0
+ // after measurement.
+ lp.width = LayoutParams.WRAP_CONTENT;
+ }
+ // Determine how big this child would like to be. If this or
+ // previous children have given a weight, then we allow it to
+ // use all available space (and we will shrink things later
+ // if needed).
+ final int usedWidth = totalWeight == 0 ? mTotalLength : 0;
+ measureChildBeforeLayout(child, i, widthMeasureSpec, usedWidth,
+ heightMeasureSpec, 0);
+ final int childWidth = child.getMeasuredWidth();
+ if (useExcessSpace) {
+ // Restore the original width and record how much space
+ // we've allocated to excess-only children so that we can
+ // match the behavior of EXACTLY measurement.
+ lp.width = 0;
+ usedExcessSpace += childWidth;
+ }
+ if (isExactly) {
+ mTotalLength += childWidth + lp.leftMargin + lp.rightMargin
+ + getNextLocationOffset(child);
+ } else {
+ final int totalLength = mTotalLength;
+ mTotalLength = Math.max(totalLength, totalLength + childWidth + lp.leftMargin
+ + lp.rightMargin + getNextLocationOffset(child));
+ }
+ if (useLargestChild) {
+ largestChildWidth = Math.max(childWidth, largestChildWidth);
+ }
+ }
+ boolean matchHeightLocally = false;
+ if (heightMode != MeasureSpec.EXACTLY && lp.height == LayoutParams.MATCH_PARENT) {
+ // The height of the linear layout will scale, and at least one
+ // child said it wanted to match our height. Set a flag indicating that
+ // we need to remeasure at least that view when we know our height.
+ matchHeight = true;
+ matchHeightLocally = true;
+ }
+ final int margin = lp.topMargin + lp.bottomMargin;
+ final int childHeight = child.getMeasuredHeight() + margin;
+ childState = combineMeasuredStates(childState, child.getMeasuredState());
+ if (baselineAligned) {
+ final int childBaseline = child.getBaseline();
+ if (childBaseline != -1) {
+ // Translates the child's vertical gravity into an index
+ // in the range 0..VERTICAL_GRAVITY_COUNT
+ final int gravity = (lp.gravity < 0 ? mGravity : lp.gravity)
+ & Gravity.VERTICAL_GRAVITY_MASK;
+ final int index = ((gravity >> Gravity.AXIS_Y_SHIFT)
+ & ~Gravity.AXIS_SPECIFIED) >> 1;
+ maxAscent[index] = Math.max(maxAscent[index], childBaseline);
+ maxDescent[index] = Math.max(maxDescent[index], childHeight - childBaseline);
+ }
+ }
+ maxHeight = Math.max(maxHeight, childHeight);
+ allFillParent = allFillParent && lp.height == LayoutParams.MATCH_PARENT;
+ if (lp.weight > 0) {
+ /*
+ * Heights of weighted Views are bogus if we end up
+ * remeasuring, so keep them separate.
+ */
+ weightedMaxHeight = Math.max(weightedMaxHeight,
+ matchHeightLocally ? margin : childHeight);
+ } else {
+ alternativeMaxHeight = Math.max(alternativeMaxHeight,
+ matchHeightLocally ? margin : childHeight);
+ }
+ i += getChildrenSkipCount(child, i);
+ }
+ if (nonSkippedChildCount > 0 && hasDividerBeforeChildAt(count)) {
+ mTotalLength += mDividerWidth;
+ }
+ // Check mMaxAscent[INDEX_TOP] first because it maps to Gravity.TOP,
+ // the most common case
+ if (maxAscent[INDEX_TOP] != -1 ||
+ maxAscent[INDEX_CENTER_VERTICAL] != -1 ||
+ maxAscent[INDEX_BOTTOM] != -1 ||
+ maxAscent[INDEX_FILL] != -1) {
+ final int ascent = Math.max(maxAscent[INDEX_FILL],
+ Math.max(maxAscent[INDEX_CENTER_VERTICAL],
+ Math.max(maxAscent[INDEX_TOP], maxAscent[INDEX_BOTTOM])));
+ final int descent = Math.max(maxDescent[INDEX_FILL],
+ Math.max(maxDescent[INDEX_CENTER_VERTICAL],
+ Math.max(maxDescent[INDEX_TOP], maxDescent[INDEX_BOTTOM])));
+ maxHeight = Math.max(maxHeight, ascent + descent);
+ }
+ if (useLargestChild &&
+ (widthMode == MeasureSpec.AT_MOST || widthMode == MeasureSpec.UNSPECIFIED)) {
+ mTotalLength = 0;
+ nonSkippedChildCount = 0;
+ for (int i = 0; i < count; ++i) {
+ final View child = getVirtualChildAt(i);
+ if (child == null) {
+ mTotalLength += measureNullChild(i);
+ continue;
+ }
+ if (child.getVisibility() == GONE) {
+ i += getChildrenSkipCount(child, i);
+ continue;
+ }
+ nonSkippedChildCount++;
+ if (hasDividerBeforeChildAt(i)) {
+ mTotalLength += mDividerWidth;
+ }
+ final LinearLayout.LayoutParams lp = (LinearLayout.LayoutParams)
+ child.getLayoutParams();
+ if (isExactly) {
+ mTotalLength += largestChildWidth + lp.leftMargin + lp.rightMargin +
+ getNextLocationOffset(child);
+ } else {
+ final int totalLength = mTotalLength;
+ mTotalLength = Math.max(totalLength, totalLength + largestChildWidth +
+ lp.leftMargin + lp.rightMargin + getNextLocationOffset(child));
+ }
+ }
+ if (nonSkippedChildCount > 0 && hasDividerBeforeChildAt(count)) {
+ mTotalLength += mDividerWidth;
+ }
+ }
+ // Add in our padding
+ mTotalLength += /*mPaddingLeft*/ 0 + /*mPaddingRight*/ 0;
+ int widthSize = mTotalLength;
+ // Check against our minimum width
+ widthSize = Math.max(widthSize, getSuggestedMinimumWidth());
+ // Reconcile our calculated size with the widthMeasureSpec
+ int widthSizeAndState = resolveSizeAndState(widthSize, widthMeasureSpec, 0);
+ widthSize = widthSizeAndState & MEASURED_SIZE_MASK;
+ // Either expand children with weight to take up available space or
+ // shrink them if they extend beyond our current bounds. If we skipped
+ // measurement on any children, we need to measure them now.
+ int remainingExcess = widthSize - mTotalLength
+ + (mAllowInconsistentMeasurement ? 0 : usedExcessSpace);
+ if (skippedMeasure
+ || ((sRemeasureWeightedChildren || remainingExcess != 0) && totalWeight > 0.0f)) {
+ float remainingWeightSum = mWeightSum > 0.0f ? mWeightSum : totalWeight;
+ maxAscent[0] = maxAscent[1] = maxAscent[2] = maxAscent[3] = -1;
+ maxDescent[0] = maxDescent[1] = maxDescent[2] = maxDescent[3] = -1;
+ maxHeight = -1;
+ mTotalLength = 0;
+ nonSkippedChildCount = 0;
+ for (int i = 0; i < count; ++i) {
+ final View child = getVirtualChildAt(i);
+ if (child == null || child.getVisibility() == View.GONE) {
+ continue;
+ }
+ nonSkippedChildCount++;
+ if (hasDividerBeforeChildAt(i)) {
+ mTotalLength += mDividerWidth;
+ }
+ final LayoutParams lp = (LayoutParams) child.getLayoutParams();
+ final float childWeight = lp.weight;
+ if (childWeight > 0) {
+ final int share = (int) (childWeight * remainingExcess / remainingWeightSum);
+ remainingExcess -= share;
+ remainingWeightSum -= childWeight;
+ final int childWidth;
+ if (mUseLargestChild && widthMode != MeasureSpec.EXACTLY) {
+ childWidth = largestChildWidth;
+ } else if (lp.width == 0 && (!mAllowInconsistentMeasurement
+ || widthMode == MeasureSpec.EXACTLY)) {
+ // This child needs to be laid out from scratch using
+ // only its share of excess space.
+ childWidth = share;
+ } else {
+ // This child had some intrinsic width to which we
+ // need to add its share of excess space.
+ childWidth = child.getMeasuredWidth() + share;
+ }
+ final int childWidthMeasureSpec = MeasureSpec.makeMeasureSpec(
+ Math.max(0, childWidth), MeasureSpec.EXACTLY);
+ final int childHeightMeasureSpec = getChildMeasureSpec(heightMeasureSpec,
+ /*mPaddingTop*/ 0 + /*mPaddingBottom*/ 0 + lp.topMargin + lp.bottomMargin,
+ lp.height);
+ child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
+ // Child may now not fit in horizontal dimension.
+ childState = combineMeasuredStates(childState,
+ child.getMeasuredState() & MEASURED_STATE_MASK);
+ }
+ if (isExactly) {
+ mTotalLength += child.getMeasuredWidth() + lp.leftMargin + lp.rightMargin +
+ getNextLocationOffset(child);
+ } else {
+ final int totalLength = mTotalLength;
+ mTotalLength = Math.max(totalLength, totalLength + child.getMeasuredWidth() +
+ lp.leftMargin + lp.rightMargin + getNextLocationOffset(child));
+ }
+ boolean matchHeightLocally = heightMode != MeasureSpec.EXACTLY &&
+ lp.height == LayoutParams.MATCH_PARENT;
+ final int margin = lp.topMargin + lp .bottomMargin;
+ int childHeight = child.getMeasuredHeight() + margin;
+ maxHeight = Math.max(maxHeight, childHeight);
+ alternativeMaxHeight = Math.max(alternativeMaxHeight,
+ matchHeightLocally ? margin : childHeight);
+ allFillParent = allFillParent && lp.height == LayoutParams.MATCH_PARENT;
+ if (baselineAligned) {
+ final int childBaseline = child.getBaseline();
+ if (childBaseline != -1) {
+ // Translates the child's vertical gravity into an index in the range 0..2
+ final int gravity = (lp.gravity < 0 ? mGravity : lp.gravity)
+ & Gravity.VERTICAL_GRAVITY_MASK;
+ final int index = ((gravity >> Gravity.AXIS_Y_SHIFT)
+ & ~Gravity.AXIS_SPECIFIED) >> 1;
+ maxAscent[index] = Math.max(maxAscent[index], childBaseline);
+ maxDescent[index] = Math.max(maxDescent[index],
+ childHeight - childBaseline);
+ }
+ }
+ }
+ if (nonSkippedChildCount > 0 && hasDividerBeforeChildAt(count)) {
+ mTotalLength += mDividerWidth;
+ }
+ // Add in our padding
+ mTotalLength += /*mPaddingLeft*/ 0 + /*mPaddingRight*/ 0;
+ // TODO: Should we update widthSize with the new total length?
+ // Check mMaxAscent[INDEX_TOP] first because it maps to Gravity.TOP,
+ // the most common case
+ if (maxAscent[INDEX_TOP] != -1 ||
+ maxAscent[INDEX_CENTER_VERTICAL] != -1 ||
+ maxAscent[INDEX_BOTTOM] != -1 ||
+ maxAscent[INDEX_FILL] != -1) {
+ final int ascent = Math.max(maxAscent[INDEX_FILL],
+ Math.max(maxAscent[INDEX_CENTER_VERTICAL],
+ Math.max(maxAscent[INDEX_TOP], maxAscent[INDEX_BOTTOM])));
+ final int descent = Math.max(maxDescent[INDEX_FILL],
+ Math.max(maxDescent[INDEX_CENTER_VERTICAL],
+ Math.max(maxDescent[INDEX_TOP], maxDescent[INDEX_BOTTOM])));
+ maxHeight = Math.max(maxHeight, ascent + descent);
+ }
+ } else {
+ alternativeMaxHeight = Math.max(alternativeMaxHeight, weightedMaxHeight);
+ // We have no limit, so make all weighted views as wide as the largest child.
+ // Children will have already been measured once.
+ if (useLargestChild && widthMode != MeasureSpec.EXACTLY) {
+ for (int i = 0; i < count; i++) {
+ final View child = getVirtualChildAt(i);
+ if (child == null || child.getVisibility() == View.GONE) {
+ continue;
+ }
+ final LinearLayout.LayoutParams lp =
+ (LinearLayout.LayoutParams) child.getLayoutParams();
+ float childExtra = lp.weight;
+ if (childExtra > 0) {
+ child.measure(
+ MeasureSpec.makeMeasureSpec(largestChildWidth, MeasureSpec.EXACTLY),
+ MeasureSpec.makeMeasureSpec(child.getMeasuredHeight(),
+ MeasureSpec.EXACTLY));
+ }
+ }
+ }
+ }
+ if (!allFillParent && heightMode != MeasureSpec.EXACTLY) {
+ maxHeight = alternativeMaxHeight;
+ }
+ maxHeight += /*mPaddingTop*/ 0 + /*mPaddingBottom*/ 0;
+ // Check against our minimum height
+ maxHeight = Math.max(maxHeight, getSuggestedMinimumHeight());
+ setMeasuredDimension(widthSizeAndState | (childState&MEASURED_STATE_MASK),
+ resolveSizeAndState(maxHeight, heightMeasureSpec,
+ (childState<Returns the number of children to skip after measuring/laying out
+ * the specified child.
+ *
+ * @param child the child after which we want to skip children
+ * @param index the index of the child after which we want to skip children
+ * @return the number of children to skip, 0 by default
+ */
+ int getChildrenSkipCount(View child, int index) {
+ return 0;
+ }
+
+ /**
+ * Returns the size (width or height) that should be occupied by a null
+ * child.
+ *
+ * @param childIndex the index of the null child
+ * @return the width or height of the child depending on the orientation
+ */
+ int measureNullChild(int childIndex) {
+ return 0;
+ }
+
+ /**
+ * Measure the child according to the parent's measure specs. This
+ * method should be overridden by subclasses to force the sizing of
+ * children. This method is called by {@link #measureVertical(int, int)} and
+ * {@link #measureHorizontal(int, int)}.
+ *
+ * @param child the child to measure
+ * @param childIndex the index of the child in this view
+ * @param widthMeasureSpec horizontal space requirements as imposed by the parent
+ * @param totalWidth extra space that has been used up by the parent horizontally
+ * @param heightMeasureSpec vertical space requirements as imposed by the parent
+ * @param totalHeight extra space that has been used up by the parent vertically
+ */
+ void measureChildBeforeLayout(View child, int childIndex,
+ int widthMeasureSpec, int totalWidth, int heightMeasureSpec,
+ int totalHeight) {
+ measureChildWithMargins(child, widthMeasureSpec, totalWidth,
+ heightMeasureSpec, totalHeight);
+ }
+
+ /**
+ * Return the location offset of the specified child. This can be used
+ * by subclasses to change the location of a given widget.
+ *
+ * @param child the child for which to obtain the location offset
+ * @return the location offset in pixels
+ */
+ int getLocationOffset(View child) {
+ return 0;
+ }
+
+ /**
+ * Return the size offset of the next sibling of the specified child.
+ * This can be used by subclasses to change the location of the widget
+ * following child.
+ *
+ * @param child the child whose next sibling will be moved
+ * @return the location offset of the next child in pixels
+ */
+ int getNextLocationOffset(View child) {
+ return 0;
+ }
+
+ @Override
+ protected void onLayout(boolean changed, int l, int t, int r, int b) {
+ if (mOrientation == VERTICAL) {
+ layoutVertical(l, t, r, b);
+ } else {
+ layoutHorizontal(l, t, r, b);
+ }
+ }
+
+ /**
+ * Position the children during a layout pass if the orientation of this
+ * LinearLayout is set to {@link #VERTICAL}.
+ *
+ * @see #getOrientation()
+ * @see #setOrientation(int)
+ * @see #onLayout(boolean, int, int, int, int)
+ * @param left
+ * @param top
+ * @param right
+ * @param bottom
+ */
+ void layoutVertical(int left, int top, int right, int bottom) {
+ final int paddingLeft = /*mPaddingLeft*/ 0;
+ int childTop;
+ int childLeft;
+ // Where right end of child should go
+ final int width = right - left;
+ int childRight = width - /*mPaddingRight*/ 0;
+ // Space available for child
+ int childSpace = width - paddingLeft - /*mPaddingRight*/ 0;
+ final int count = getVirtualChildCount();
+ final int majorGravity = mGravity & Gravity.VERTICAL_GRAVITY_MASK;
+ final int minorGravity = mGravity & Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK;
+ switch (majorGravity) {
+ case Gravity.BOTTOM:
+ // mTotalLength contains the padding already
+ childTop = /*mPaddingTop*/ 0 + bottom - top - mTotalLength;
+ break;
+ // mTotalLength contains the padding already
+ case Gravity.CENTER_VERTICAL:
+ childTop = /*mPaddingTop*/ 0 + (bottom - top - mTotalLength) / 2;
+ break;
+ case Gravity.TOP:
+ default:
+ childTop = /*mPaddingTop*/ 0;
+ break;
+ }
+ for (int i = 0; i < count; i++) {
+ final View child = getVirtualChildAt(i);
+ if (child == null) {
+ childTop += measureNullChild(i);
+ } else if (child.getVisibility() != GONE) {
+ final int childWidth = child.getMeasuredWidth();
+ final int childHeight = child.getMeasuredHeight();
+ final LinearLayout.LayoutParams lp =
+ (LinearLayout.LayoutParams) child.getLayoutParams();
+ int gravity = lp.gravity;
+ if (gravity < 0) {
+ gravity = minorGravity;
+ }
+ final int layoutDirection = getLayoutDirection();
+ final int absoluteGravity = Gravity.getAbsoluteGravity(gravity, layoutDirection);
+ switch (absoluteGravity & Gravity.HORIZONTAL_GRAVITY_MASK) {
+ case Gravity.CENTER_HORIZONTAL:
+ childLeft = paddingLeft + ((childSpace - childWidth) / 2)
+ + lp.leftMargin - lp.rightMargin;
+ break;
+ case Gravity.RIGHT:
+ childLeft = childRight - childWidth - lp.rightMargin;
+ break;
+ case Gravity.LEFT:
+ default:
+ childLeft = paddingLeft + lp.leftMargin;
+ break;
+ }
+ if (hasDividerBeforeChildAt(i)) {
+ childTop += mDividerHeight;
+ }
+ childTop += lp.topMargin;
+ setChildFrame(child, childLeft, childTop + getLocationOffset(child),
+ childWidth, childHeight);
+ childTop += childHeight + lp.bottomMargin + getNextLocationOffset(child);
+ i += getChildrenSkipCount(child, i);
+ }
+ }
+ }
+
+ /**
+ * Position the children during a layout pass if the orientation of this
+ * LinearLayout is set to {@link #HORIZONTAL}.
+ *
+ * @see #getOrientation()
+ * @see #setOrientation(int)
+ * @see #onLayout(boolean, int, int, int, int)
+ * @param left
+ * @param top
+ * @param right
+ * @param bottom
+ */
+ void layoutHorizontal(int left, int top, int right, int bottom) {
+ final boolean isLayoutRtl = /*isLayoutRtl()*/ false;
+ final int paddingTop = /*mPaddingTop*/ 0;
+ int childTop;
+ int childLeft;
+ // Where bottom of child should go
+ final int height = bottom - top;
+ int childBottom = height - /*mPaddingBottom*/ 0;
+ // Space available for child
+ int childSpace = height - paddingTop - /*mPaddingBottom*/ 0;
+ final int count = getVirtualChildCount();
+ final int majorGravity = mGravity & Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK;
+ final int minorGravity = mGravity & Gravity.VERTICAL_GRAVITY_MASK;
+ final boolean baselineAligned = mBaselineAligned;
+ final int[] maxAscent = mMaxAscent;
+ final int[] maxDescent = mMaxDescent;
+ final int layoutDirection = getLayoutDirection();
+ switch (Gravity.getAbsoluteGravity(majorGravity, layoutDirection)) {
+ case Gravity.RIGHT:
+ // mTotalLength contains the padding already
+ childLeft = /*mPaddingLeft*/ 0 + right - left - mTotalLength;
+ break;
+ case Gravity.CENTER_HORIZONTAL:
+ // mTotalLength contains the padding already
+ childLeft = /*mPaddingLeft*/ 0 + (right - left - mTotalLength) / 2;
+ break;
+ case Gravity.LEFT:
+ default:
+ childLeft = /*mPaddingLeft*/ 0;
+ break;
+ }
+ int start = 0;
+ int dir = 1;
+ //In case of RTL, start drawing from the last child.
+ if (isLayoutRtl) {
+ start = count - 1;
+ dir = -1;
+ }
+ for (int i = 0; i < count; i++) {
+ final int childIndex = start + dir * i;
+ final View child = getVirtualChildAt(childIndex);
+ if (child == null) {
+ childLeft += measureNullChild(childIndex);
+ } else if (child.getVisibility() != GONE) {
+ final int childWidth = child.getMeasuredWidth();
+ final int childHeight = child.getMeasuredHeight();
+ int childBaseline = -1;
+ final LinearLayout.LayoutParams lp =
+ (LinearLayout.LayoutParams) child.getLayoutParams();
+ if (baselineAligned && lp.height != LayoutParams.MATCH_PARENT) {
+ childBaseline = child.getBaseline();
+ }
+ int gravity = lp.gravity;
+ if (gravity < 0) {
+ gravity = minorGravity;
+ }
+ switch (gravity & Gravity.VERTICAL_GRAVITY_MASK) {
+ case Gravity.TOP:
+ childTop = paddingTop + lp.topMargin;
+ if (childBaseline != -1) {
+ childTop += maxAscent[INDEX_TOP] - childBaseline;
+ }
+ break;
+ case Gravity.CENTER_VERTICAL:
+ // Removed support for baseline alignment when layout_gravity or
+ // gravity == center_vertical. See bug #1038483.
+ // Keep the code around if we need to re-enable this feature
+ // if (childBaseline != -1) {
+ // // Align baselines vertically only if the child is smaller than us
+ // if (childSpace - childHeight > 0) {
+ // childTop = paddingTop + (childSpace / 2) - childBaseline;
+ // } else {
+ // childTop = paddingTop + (childSpace - childHeight) / 2;
+ // }
+ // } else {
+ childTop = paddingTop + ((childSpace - childHeight) / 2)
+ + lp.topMargin - lp.bottomMargin;
+ break;
+ case Gravity.BOTTOM:
+ childTop = childBottom - childHeight - lp.bottomMargin;
+ if (childBaseline != -1) {
+ int descent = child.getMeasuredHeight() - childBaseline;
+ childTop -= (maxDescent[INDEX_BOTTOM] - descent);
+ }
+ break;
+ default:
+ childTop = paddingTop;
+ break;
+ }
+ if (isLayoutRtl) {
+ // Because rtl rendering occurs in the reverse direction, we need to check
+ // after the child rather than before (since after=left in this context)
+ if (hasDividerAfterChildAt(childIndex)) {
+ childLeft += mDividerWidth;
+ }
+ } else if (hasDividerBeforeChildAt(childIndex)) {
+ childLeft += mDividerWidth;
+ }
+ childLeft += lp.leftMargin;
+ setChildFrame(child, childLeft + getLocationOffset(child), childTop,
+ childWidth, childHeight);
+ childLeft += childWidth + lp.rightMargin +
+ getNextLocationOffset(child);
+ i += getChildrenSkipCount(child, childIndex);
+ }
+ }
+ }
+
+ private void setChildFrame(View child, int left, int top, int width, int height) {
+ child.layout(left, top, left + width, top + height);
+ }
+
+ /**
+ * Should the layout be a column or a row.
+ * @param orientation Pass {@link #HORIZONTAL} or {@link #VERTICAL}. Default
+ * value is {@link #HORIZONTAL}.
+ *
+ * @attr ref android.R.styleable#LinearLayout_orientation
+ */
+ public void setOrientation(@OrientationMode int orientation) {
+ if (mOrientation != orientation) {
+ mOrientation = orientation;
+ requestLayout();
+ }
+ }
+
+ /**
+ * Returns the current orientation.
+ *
+ * @return either {@link #HORIZONTAL} or {@link #VERTICAL}
+ */
+ @OrientationMode
+ public int getOrientation() {
+ return mOrientation;
+ }
+
+ /**
+ * Describes how the child views are positioned. Defaults to GRAVITY_TOP. If
+ * this layout has a VERTICAL orientation, this controls where all the child
+ * views are placed if there is extra vertical space. If this layout has a
+ * HORIZONTAL orientation, this controls the alignment of the children.
+ *
+ * @param gravity See {@link android.view.Gravity}
+ *
+ * @attr ref android.R.styleable#LinearLayout_gravity
+ */
+ public void setGravity(int gravity) {
+ if (mGravity != gravity) {
+ if ((gravity & Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK) == 0) {
+ gravity |= Gravity.START;
+ }
+ if ((gravity & Gravity.VERTICAL_GRAVITY_MASK) == 0) {
+ gravity |= Gravity.TOP;
+ }
+ mGravity = gravity;
+ requestLayout();
+ }
+ }
+
+ /**
+ * Returns the current gravity. See {@link android.view.Gravity}
+ *
+ * @return the current gravity.
+ * @see #setGravity
+ */
+ public int getGravity() {
+ return mGravity;
+ }
+
+ public void setHorizontalGravity(int horizontalGravity) {
+ final int gravity = horizontalGravity & Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK;
+ if ((mGravity & Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK) != gravity) {
+ mGravity = (mGravity & ~Gravity.RELATIVE_HORIZONTAL_GRAVITY_MASK) | gravity;
+ requestLayout();
+ }
+ }
+
+ public void setVerticalGravity(int verticalGravity) {
+ final int gravity = verticalGravity & Gravity.VERTICAL_GRAVITY_MASK;
+ if ((mGravity & Gravity.VERTICAL_GRAVITY_MASK) != gravity) {
+ mGravity = (mGravity & ~Gravity.VERTICAL_GRAVITY_MASK) | gravity;
+ requestLayout();
+ }
}
@Override
public LayoutParams generateLayoutParams(AttributeSet attrs) {
- return new LayoutParams(getContext(), attrs);
+ return new LinearLayout.LayoutParams(getContext(), attrs);
}
+ /**
+ * Returns a set of layout parameters with a width of
+ * {@link android.view.ViewGroup.LayoutParams#MATCH_PARENT}
+ * and a height of {@link android.view.ViewGroup.LayoutParams#WRAP_CONTENT}
+ * when the layout's orientation is {@link #VERTICAL}. When the orientation is
+ * {@link #HORIZONTAL}, the width is set to {@link LayoutParams#WRAP_CONTENT}
+ * and the height to {@link LayoutParams#WRAP_CONTENT}.
+ */
+ @Override
+ protected LayoutParams generateDefaultLayoutParams() {
+ if (mOrientation == HORIZONTAL) {
+ return new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT);
+ } else if (mOrientation == VERTICAL) {
+ return new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.WRAP_CONTENT);
+ }
+ return null;
+ }
+
+ @Override
+ protected LayoutParams generateLayoutParams(ViewGroup.LayoutParams lp) {
+ if (/*sPreserveMarginParamsInLayoutParamConversion*/ true) {
+ if (lp instanceof LayoutParams) {
+ return new LayoutParams((LayoutParams) lp);
+ } else if (lp instanceof MarginLayoutParams) {
+ return new LayoutParams((MarginLayoutParams) lp);
+ }
+ }
+ return new LayoutParams(lp);
+ }
+
+ // Override to allow type-checking of LayoutParams.
+ @Override
+ protected boolean checkLayoutParams(ViewGroup.LayoutParams p) {
+ return p instanceof LinearLayout.LayoutParams;
+ }
+
+ public CharSequence getAccessibilityClassName() {
+ return LinearLayout.class.getName();
+ }
+
+ /**
+ * Per-child layout information associated with ViewLinearLayout.
+ *
+ * @attr ref android.R.styleable#LinearLayout_Layout_layout_weight
+ * @attr ref android.R.styleable#LinearLayout_Layout_layout_gravity
+ */
public static class LayoutParams extends ViewGroup.MarginLayoutParams {
- public LayoutParams (Context c, AttributeSet attrs) {
+ /**
+ * Indicates how much of the extra space in the LinearLayout will be
+ * allocated to the view associated with these LayoutParams. Specify
+ * 0 if the view should not be stretched. Otherwise the extra pixels
+ * will be pro-rated among all views whose weight is greater than 0.
+ */
+ public float weight;
+ /**
+ * Gravity for the view associated with these LayoutParams.
+ *
+ * @see android.view.Gravity
+ */
+ public int gravity = -1;
+
+ /**
+ * {@inheritDoc}
+ */
+ public LayoutParams(Context c, AttributeSet attrs) {
super(c, attrs);
+ TypedArray a =
+ c.obtainStyledAttributes(attrs, com.android.internal.R.styleable.LinearLayout_Layout);
+ weight = a.getFloat(com.android.internal.R.styleable.LinearLayout_Layout_layout_weight, 0);
+ gravity = a.getInt(com.android.internal.R.styleable.LinearLayout_Layout_layout_gravity, -1);
+ a.recycle();
}
+ /**
+ * {@inheritDoc}
+ */
public LayoutParams(int width, int height) {
super(width, height);
+ weight = 0;
}
+ /**
+ * Creates a new set of layout parameters with the specified width, height
+ * and weight.
+ *
+ * @param width the width, either {@link #MATCH_PARENT},
+ * {@link #WRAP_CONTENT} or a fixed size in pixels
+ * @param height the height, either {@link #MATCH_PARENT},
+ * {@link #WRAP_CONTENT} or a fixed size in pixels
+ * @param weight the weight
+ */
public LayoutParams(int width, int height, float weight) {
- super(width, height, weight);
+ super(width, height);
+ this.weight = weight;
}
+
+ /**
+ * {@inheritDoc}
+ */
+ public LayoutParams(ViewGroup.LayoutParams p) {
+ super(p);
+ }
+
+ /**
+ * {@inheritDoc}
+ */
+ public LayoutParams(ViewGroup.MarginLayoutParams source) {
+ super(source);
+ }
+
+ /**
+ * Copy constructor. Clones the width, height, margin values, weight,
+ * and gravity of the source.
+ *
+ * @param source The layout params to copy from.
+ */
+ public LayoutParams(LayoutParams source) {
+ super(source);
+ this.weight = source.weight;
+ this.gravity = source.gravity;
+ }
+
}
}