package android.view; import android.R; import android.animation.LayoutTransition; import android.content.Context; import android.content.res.TypedArray; import android.util.AttributeSet; import java.util.ArrayList; import java.util.Iterator; public class ViewGroup extends View implements ViewParent, ViewManager { public ArrayList children; public ViewGroup(Context context) { this(context, null); } public ViewGroup(Context context, AttributeSet attrs) { this(context, attrs, 0); } public ViewGroup(Context context, AttributeSet attrs, int defStyleAttr) { this(context, attrs, defStyleAttr, 0); } public ViewGroup(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { super(context, attrs, defStyleAttr); children = new ArrayList(); haveComplexMeasure = true; } public void addView(View child) { addView(child, -1); } public void addView(View child, int index) { LayoutParams params = child.getLayoutParams(); if (params == null) { params = generateDefaultLayoutParams(); } addView(child, index, params); } public void addView(View child, LayoutParams params) { addView(child, -1, params); } public void addView(View child, int width, int height) { final LayoutParams params = generateDefaultLayoutParams(); params.width = width; params.height = height; addView(child, params); } protected void addViewInternal(View child, int index, LayoutParams params) { if (child.parent == this) return; if (!checkLayoutParams(params)) { params = generateLayoutParams(params); } child.parent = this; child.setLayoutParams(params); if (index < 0) index = children.size(); children.add(index, child); native_addView(widget, child.widget, index, params); if (isAttachedToWindow()) { child.onAttachedToWindow(); } requestLayout(); } public void addView(View child, int index, LayoutParams params) { addViewInternal(child, index, params); } protected boolean addViewInLayout(View child, int index, LayoutParams params) { addViewInternal(child, index, params); return true; } public void removeView(View child) { if (child.parent != this) return; child.parent = null; children.remove(child); native_removeView(widget, child.widget); } public void removeViewAt(int index) { removeView(children.get(index)); } public void removeAllViews() { for (Iterator it = children.iterator(); it.hasNext();) { View child = it.next(); child.parent = null; it.remove(); native_removeView(widget, child.widget); } } public void detachViewFromParent(int index) { } public void attachViewToParent(View view, int index, LayoutParams params) { addViewInternal(view, index, params); } protected void removeDetachedView(View child, boolean animate) { removeView(child); } protected native void native_addView(long widget, long child, int index, LayoutParams params); protected native void native_removeView(long widget, long child); public View getChildAt(int index) { try { return children.get(index); } catch (IndexOutOfBoundsException e) { return null; } } public int indexOfChild(View child) { return children.indexOf(child); } public void updateViewLayout(View view, ViewGroup.LayoutParams params) {} public LayoutParams generateLayoutParams(AttributeSet attrs) { return new LayoutParams(getContext(), attrs); } protected LayoutParams generateLayoutParams(ViewGroup.LayoutParams p) { return p; } public void bringChildToFront(View child) { // TODO: actually implement this (might make sense to implement it in the subclasses instead), when applicable } /** * Returns the number of children in the group. * * @return a positive integer representing the number of children in * the group */ public int getChildCount() { return children.size(); } public void setMotionEventSplittingEnabled(boolean enabled) {} public void setOnHierarchyChangeListener(OnHierarchyChangeListener listener) {} protected boolean checkLayoutParams(LayoutParams params) { return true; } public LayoutTransition getLayoutTransition() {return null;} public static int getChildMeasureSpec(int spec, int padding, int childDimension) { int specMode = MeasureSpec.getMode(spec); int specSize = MeasureSpec.getSize(spec); int size = Math.max(0, specSize - padding); int resultSize = 0; int resultMode = 0; switch (specMode) { // Parent has imposed an exact size on us case MeasureSpec.EXACTLY: if (childDimension >= 0) { resultSize = childDimension; resultMode = MeasureSpec.EXACTLY; } else if (childDimension == LayoutParams.MATCH_PARENT) { // Child wants to be our size. So be it. resultSize = size; resultMode = MeasureSpec.EXACTLY; } else if (childDimension == LayoutParams.WRAP_CONTENT) { // Child wants to determine its own size. It can't be // bigger than us. resultSize = size; resultMode = MeasureSpec.AT_MOST; } break; // Parent has imposed a maximum size on us case MeasureSpec.AT_MOST: if (childDimension >= 0) { // Child wants a specific size... so be it resultSize = childDimension; resultMode = MeasureSpec.EXACTLY; } else if (childDimension == LayoutParams.MATCH_PARENT) { // Child wants to be our size, but our size is not fixed. // Constrain child to not be bigger than us. resultSize = size; resultMode = MeasureSpec.AT_MOST; } else if (childDimension == LayoutParams.WRAP_CONTENT) { // Child wants to determine its own size. It can't be // bigger than us. resultSize = size; resultMode = MeasureSpec.AT_MOST; } break; // Parent asked to see how big we want to be case MeasureSpec.UNSPECIFIED: if (childDimension >= 0) { // Child wants a specific size... let them have it resultSize = childDimension; resultMode = MeasureSpec.EXACTLY; } else if (childDimension == LayoutParams.MATCH_PARENT) { // Child wants to be our size... find out how big it should // be resultSize = 0; //View.sUseZeroUnspecifiedMeasureSpec ? 0 : size; resultMode = MeasureSpec.UNSPECIFIED; } else if (childDimension == LayoutParams.WRAP_CONTENT) { // Child wants to determine its own size.... find out how // big it should be resultSize = 0; //View.sUseZeroUnspecifiedMeasureSpec ? 0 : size; resultMode = MeasureSpec.UNSPECIFIED; } break; } //noinspection ResourceType return MeasureSpec.makeMeasureSpec(resultSize, resultMode); } protected void measureChildWithMargins(View child, int parentWidthMeasureSpec, int widthUsed, int parentHeightMeasureSpec, int heightUsed) { final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams(); final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec, /*mPaddingLeft + mPaddingRight +*/ lp.leftMargin + lp.rightMargin + widthUsed, lp.width); final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec, /*mPaddingTop + mPaddingBottom +*/ lp.topMargin + lp.bottomMargin + heightUsed, lp.height); child.measure(childWidthMeasureSpec, childHeightMeasureSpec); } protected void measureChild(View child, int parentWidthMeasureSpec, int parentHeightMeasureSpec) { final LayoutParams lp = child.getLayoutParams(); final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec, /*mPaddingLeft + mPaddingRight*/0, lp.width); final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec, /*mPaddingTop + mPaddingBottom*/0, lp.height); child.measure(childWidthMeasureSpec, childHeightMeasureSpec); } public void setAddStatesFromChildren(boolean addsStates) {} public View getFocusedChild() {return null;} public int getDescendantFocusability() {return 0;} public void startViewTransition(View view) {} public void endViewTransition(View view) {} protected LayoutParams generateDefaultLayoutParams() { return new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT); } public void focusableViewAvailable(View v) {} @Override public void setGravity(int gravity) { super.setGravity(gravity); // update children as necessary for (View child: children) { LayoutParams params = child.getLayoutParams(); if (params.gravity == -1) child.setLayoutParams(params); } } protected void setChildrenDrawingOrderEnabled(boolean enabled) {} public void requestDisallowInterceptTouchEvent(boolean disallowIntercept) {} @Override protected void onAttachedToWindow() { super.onAttachedToWindow(); for (View child: children) { child.onAttachedToWindow(); } } protected boolean isChildrenDrawingOrderEnabled() { return false; } @Override public boolean onStartNestedScroll(View child, View target, int nestedScrollAxes) { return false; } @Override public boolean onNestedPreFling(View target, float velocityX, float velocityY) { return false; } @Override public boolean onNestedFling(View target, float velocityX, float velocityY, boolean consumed) { return false; } public void setClipToPadding(boolean clipToPadding) {} public static class LayoutParams { public static final int FILL_PARENT = -1; public static final int MATCH_PARENT = -1; public static final int WRAP_CONTENT = -2; public int width = 0; public int height = 0; public float weight = 0; public int gravity = -1; public LayoutParams() { // FIXME } public LayoutParams(int width, int height) { this.width = width; this.height = height; } public LayoutParams(int width, int height, float weight) { this.width = width; this.height = height; this.weight = weight; } public LayoutParams(Context context, AttributeSet attrs) { TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.ViewGroup_Layout); width = a.getLayoutDimension(R.styleable.ViewGroup_Layout_layout_width, "layout_width"); height = a.getLayoutDimension(R.styleable.ViewGroup_Layout_layout_height, "layout_height"); this.gravity = attrs.getAttributeIntValue("http://schemas.android.com/apk/res/android", "layout_gravity", -1); a.recycle(); } public void resolveLayoutDirection(int layoutDirection) {} /** * Used to animate layouts. */ // public LayoutAnimationController.AnimationParameters layoutAnimationParameters; } public static class MarginLayoutParams extends ViewGroup.LayoutParams { public int leftMargin; public int topMargin; public int rightMargin; public int bottomMargin; public MarginLayoutParams() { super(); } public MarginLayoutParams(LayoutParams params) { super(); width = params.width; height = params.height; } public MarginLayoutParams(int width, int height) { super(width, height); } public MarginLayoutParams(int width, int height, float weight) { super(width, height, weight); } public MarginLayoutParams(Context context, AttributeSet attributeSet) { super(context, attributeSet); TypedArray a = context.obtainStyledAttributes(attributeSet, R.styleable.ViewGroup_MarginLayout); leftMargin = a.getDimensionPixelSize(R.styleable.ViewGroup_MarginLayout_layout_marginLeft, 0); topMargin = a.getDimensionPixelSize(R.styleable.ViewGroup_MarginLayout_layout_marginTop, 0); rightMargin = a.getDimensionPixelSize(R.styleable.ViewGroup_MarginLayout_layout_marginRight, 0); bottomMargin = a.getDimensionPixelSize(R.styleable.ViewGroup_MarginLayout_layout_marginBottom, 0); a.recycle(); } public void setMargins(int left, int top, int right, int bottom) { leftMargin = left; topMargin = top; rightMargin = right; bottomMargin = bottom; } public int getMarginStart() { return leftMargin; } public int getMarginEnd() { return rightMargin; } public void setMarginStart(int marginStart) { leftMargin = marginStart; } public void setMarginEnd(int marginEnd) { rightMargin = marginEnd; } public int getLayoutDirection() { return LAYOUT_DIRECTION_LTR; } } public interface OnHierarchyChangeListener { } }