Android API:自定义ViewGroup

来源:互联网 发布:张兆艺淘宝店 编辑:程序博客网 时间:2024/06/10 17:22
 无论是出于项目功能需求还是出于界面美观,或者是因为项目中对控件进行封装等,自定义View在我们开发中是非常常见的, Android官方DEMO中也提供了自定义View的例子,今天学习的是自定义ViewGroup.
public class CustomLayout extends ViewGroup {    /** 子控件离左边的距离 */    private int mLeftWidth;    /** 子控件离左边的距离 */    private int mRightWidth;    /** 计算子控件容器的范围 */    private final Rect mTmpContainerRect = new Rect();    /** 计算子控件的范围 */    private final Rect mTmpChildRect = new Rect();    /*     * View的三个构造函数     */    public CustomLayout(Context context) {        super(context);    }    public CustomLayout(Context context, AttributeSet attrs) {        this(context, attrs, 0);    }    public CustomLayout(Context context, AttributeSet attrs, int defStyle) {        super(context, attrs, defStyle);    }    /**     * 任何不滚动的布局管理器都应该像这样重写该方法     */    @Override    public boolean shouldDelayChildPressedState() {        return false;    }    /**     * 测量(子)控件大小     */    @Override    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {        // 子控件数量        int count = getChildCount();        mLeftWidth = 0;        mRightWidth = 0;        // 设置最大高度,最大宽度及状态        int maxHeight = 0;        int maxWidth = 0;        int childState = 0;        // 遍历所有子控件,分别进行测量        for (int i = 0; i < count; i++) {            final View child = getChildAt(i);            if (child.getVisibility() != GONE) {                // 测量子控件:某一个子view宽及高,                // 内部加上了viewGroup的padding值、margin值和传入的宽高widthUsed、heightUsed                measureChildWithMargins(child, widthMeasureSpec, 0,                        heightMeasureSpec, 0);                // 通过LayoutParams来设置大小                final LayoutParams lp = (LayoutParams) child.getLayoutParams();                if (lp.position == LayoutParams.POSITION_LEFT) {                    mLeftWidth += Math.max(maxWidth, child.getMeasuredWidth()                            + lp.leftMargin + lp.rightMargin);                } else if (lp.position == LayoutParams.POSITION_RIGHT) {                    mRightWidth += Math.max(maxWidth, child.getMeasuredWidth()                            + lp.leftMargin + lp.rightMargin);                } else {                    maxWidth = Math.max(maxWidth, child.getMeasuredWidth()                            + lp.leftMargin + lp.rightMargin);                }                maxHeight = Math.max(maxHeight, child.getMeasuredHeight()                        + lp.topMargin + lp.bottomMargin);                childState = combineMeasuredStates(childState,                        child.getMeasuredState());            }        }        // 总宽度是内部所有子视图的宽度加上其他约束的宽度        maxWidth += mLeftWidth + mRightWidth;        // 检查默认的最小高度和最小宽度,取最大值        maxHeight = Math.max(maxHeight, getSuggestedMinimumHeight());        maxWidth = Math.max(maxWidth, getSuggestedMinimumWidth());        // 设置最终的尺寸        setMeasuredDimension(                resolveSizeAndState(maxWidth, widthMeasureSpec, childState),                resolveSizeAndState(maxHeight, heightMeasureSpec,                        childState << MEASURED_HEIGHT_STATE_SHIFT));    }    /**     * 设置子控件在ViewGroup中的位置     */    @Override    protected void onLayout(boolean changed, int left, int top, int right,            int bottom) {        final int count = getChildCount();        // 视图在布局中放置的左右外边        int leftPos = getPaddingLeft();        int rightPos = right - left - getPaddingRight();        // 视图流内部的中心区域        final int middleLeft = leftPos + mLeftWidth;        final int middleRight = rightPos - mRightWidth;        // 所操作的布局的顶边和底边        final int parentTop = getPaddingTop();        final int parentBottom = bottom - top - getPaddingBottom();        for (int i = 0; i < count; i++) {            final View child = getChildAt(i);            if (child.getVisibility() != GONE) {                final LayoutParams lp = (LayoutParams) child.getLayoutParams();                // 测量宽度和测量高度                final int width = child.getMeasuredWidth();                final int height = child.getMeasuredHeight();                // 计算正在放置子视图的窗体                if (lp.position == LayoutParams.POSITION_LEFT) {                    mTmpContainerRect.left = leftPos + lp.leftMargin;                    mTmpContainerRect.right = leftPos + width + lp.rightMargin;                    leftPos = mTmpContainerRect.right;                } else if (lp.position == LayoutParams.POSITION_RIGHT) {                    mTmpContainerRect.right = rightPos - lp.rightMargin;                    mTmpContainerRect.left = rightPos - width - lp.leftMargin;                    rightPos = mTmpContainerRect.left;                } else {                    mTmpContainerRect.left = middleLeft + lp.leftMargin;                    mTmpContainerRect.right = middleRight - lp.rightMargin;                }                mTmpContainerRect.top = parentTop + lp.topMargin;                mTmpContainerRect.bottom = parentBottom - lp.bottomMargin;                // //通过子视图的重心值和尺寸决定其在容器内的最终布局                Gravity.apply(lp.gravity, width, height, mTmpContainerRect,                        mTmpChildRect);                // 放置子视图                child.layout(mTmpChildRect.left, mTmpChildRect.top,                        mTmpChildRect.right, mTmpChildRect.bottom);            }        }    }    /**     * 下面的实现部分是针对每个子视图的布局参数的,如果你不需要这些(比如说你写了一个布局管理器)不需要混合放置子视图,那么你可以删除这部分     */    // ----------------------------------------------------------------------    /**     * 要自定义ViewGroup支持子控件的layout_margin参数,     * 则自定义的ViewGroup类必须重载generateLayoutParams     * ()函数,并且在该函数中返回一个ViewGroup.MarginLayoutParams派生类对象,这样才能使用margin参数。     */    @Override    public LayoutParams generateLayoutParams(AttributeSet attrs) {        return new CustomLayout.LayoutParams(getContext(), attrs);    }    // 返回一组默认布局参数。没有设置布局参数执行addview(View)时,这些参数被请求时返回null,并抛出一个异常。    @Override    protected LayoutParams generateDefaultLayoutParams() {        return new LayoutParams(LayoutParams.MATCH_PARENT,                LayoutParams.MATCH_PARENT);    }    @Override    protected ViewGroup.LayoutParams generateLayoutParams(            ViewGroup.LayoutParams p) {        return new LayoutParams(p);    }    @Override    protected boolean checkLayoutParams(ViewGroup.LayoutParams p) {        return p instanceof LayoutParams;    }    /**     * 自定义子控件的LayoutParams     */    public static class LayoutParams extends MarginLayoutParams {        // 设置居中方式        public int gravity = Gravity.TOP | Gravity.START;        public static int POSITION_MIDDLE = 0;        public static int POSITION_LEFT = 1;        public static int POSITION_RIGHT = 2;        public int position = POSITION_MIDDLE;        public LayoutParams(Context c, AttributeSet attrs) {            super(c, attrs);            // 获取定义在XML中的自定义属性            TypedArray a = c.obtainStyledAttributes(attrs,                    R.styleable.CustomLayoutLP);            gravity = a.getInt(                    R.styleable.CustomLayoutLP_android_layout_gravity, gravity);            position = a.getInt(R.styleable.CustomLayoutLP_layout_position,                    position);            a.recycle();        }        public LayoutParams(int width, int height) {            super(width, height);        }        public LayoutParams(ViewGroup.LayoutParams source) {            super(source);        }    }}
 把定义好的ViewGroup当作我们通常使用的诸如LinearLayout等控件一样使用就可以啦。

更多API代码:https://github.com/ldm520/ANDROID_API_DEMOS等Android官方API开源代码。

0 0
原创粉丝点击