Android - 居中的FlowLayout

前言

因为需求的原因,需要去使用流式布局,但是这次我们的需求,和我之前的见到的流式布局不太一样。因为我们的是居中显示的流式布局。这时候,就得自己去自定义了。

老规矩,先看图。


这里说一下,我的实现思路:
第一种添加子View的方式:就是通过addView(view)添加
① 这里需要自定义两个属性mChildSpacing和mRowSpacing,一个是用来控制child与child之间的间距(marginRight),一个是用来控制child的marginTop值

第二种是在xml中直接向ViewGroup中添加子View
① 直接添加子View,我们需要去设置子View的marginRight和marginTop,marginBottom也可以设置。不需要去设置marginLeft了。

既然要居中显示,就需要减去父View的paddingLeft和paddingRight值,将(剩余的宽度-该行控件的全部宽度)/2,这时候就均分了左右两边的剩余宽度了。我们就可以用父View的paddingLeft+均分后的值,就是第一个控件的初始left值。然后后面的控件依次向后排列即可。

到控件占满一行时,就需要换行了,这时候,curTop(父View paddingTop)+上一行中最大子View的高度+mRowSpacing(marginTop)得到的值就是另起一行的top值

大概就是这个思路了,下面我画了一个草图:


下面是自定义ViewGroup的全部代码和详细注释:

public class CenterFlowLayout extends ViewGroup {

    //默认item的spacing
    private static final int DEFAULT_CHILD_SPACING = 0;
    //默认item的topMargin值
    private static final int DEFAULT_ROW_SPACING = 0;
    //子View之间的间距
    private int mChildSpacing = DEFAULT_CHILD_SPACING;
    //子View的marginTop值
    private int mRowSpacing = DEFAULT_ROW_SPACING;
    //用来存储每行item所占用的宽度的总和
    private List<Integer> itemLineWidth = new ArrayList<>();
    //用来存储每行item的个数
    private List<Integer> itemLineNum = new ArrayList<>();
    //用来记录item所占用的总行数
    private int mRowTotalCount = 0;

    public CenterFlowLayout(Context context) {
        this(context, null);
    }

    public CenterFlowLayout(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public CenterFlowLayout(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);

        TypedArray typedArray = context.getTheme().obtainStyledAttributes(attrs, R.styleable.CenterFlowLayout, 0, 0);
        mChildSpacing = typedArray.getDimensionPixelSize(R.styleable.CenterFlowLayout_childSpacing, DEFAULT_CHILD_SPACING);
        mRowSpacing = typedArray.getDimensionPixelSize(R.styleable.CenterFlowLayout_rowSpacing, DEFAULT_ROW_SPACING);
        typedArray.recycle();
    }

    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        //获取父View的paddingLeft值
        int childLeft = getPaddingLeft();
        //获取父View的paddingTop值
        int childTop = getPaddingTop();
        //获取去除paddingRight之后的宽度
        int childRight = getMeasuredWidth() - getPaddingRight();
        //获取实际子View可用的宽度
        int availableWidth = childRight - childLeft;
        //子View初始时left值
        int curLeft;
        //子View初始时的top值
        int curTop = childTop;
        //单行中最大子View的高度
        int maxHeight = 0;
        //子View的高度
        int childHeight;
        //子View宽度
        int childWidth;
        //父View中子View的index
        int childIndex = 0;
        for (int j = 0; j < mRowTotalCount; j++) {
            //获取单行中子View的个数
            Integer childNum = itemLineNum.get(j);
            //初始化子View的left值
            curLeft = childLeft + (availableWidth - itemLineWidth.get(j)) / 2;
            //竖直方向上的margin值
            int verticalMargin = 0;
            for (int i = 0; i < childNum; i++) {
                //获取ViewGroup中的子View
                View child = getChildAt(childIndex++);
                //跳过可见性为GONE的子View
                if (child.getVisibility() == View.GONE) {
                    continue;
                }
                //获取子View的宽度
                childWidth = child.getMeasuredWidth();
                //获取子View的高度
                childHeight = child.getMeasuredHeight();
                //获取子View的LayoutParams
                MarginLayoutParams params = (CenterLayoutParams) child.getLayoutParams();
                int marginRight = 0, marginTop = 0, marginBottom;
                if (params instanceof MarginLayoutParams) {
                    //获取子View的marginRight值
                    marginRight = params.rightMargin;
                    //获取子View的marginTop值
                    marginTop = params.topMargin;
                    //获取子View的marginBottom值
                    marginBottom = params.bottomMargin;
                    //获取子View上下间距的和
                    if (childNum > 1 && i == 0) {
                        verticalMargin = marginTop + marginBottom;
                    }
                }
                //对子View进行布局
                child.layout(curLeft, curTop, curLeft + childWidth, curTop + childHeight);
                //找到单行中子View的最高高度
                if (maxHeight < childHeight) {
                    maxHeight = childHeight;
                }
                //叠加left值,向右一次排列
                curLeft += childWidth + mChildSpacing + marginRight;
            }
            //换行时的高度
            curTop += maxHeight + mRowSpacing + verticalMargin;
            //最大高度重置
            maxHeight = 0;
        }

    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        //将属性值重置
        itemLineNum.clear();
        itemLineWidth.clear();
        mRowTotalCount = 0;
        //获取ViewGroup的宽度
        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        //获取ViewGroup的宽度mode
        int widthMode = MeasureSpec.getMode(widthMeasureSpec);
        //获取ViewGroup的高度
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);
        //获取ViewGroup的高度mode
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);
        //获取ViewGroup的可用宽度
        int rowLength = widthSize - getPaddingLeft() - getPaddingRight();
        //测量的宽度
        int measuredWidth = 0;
        //测量的高度
        int measuredHeight = 0;
        //子View最大的宽度
        int maxWidth = 0;
        //子View最大的高度
        int maxHeight = 0;
        //子View的总行数
        int rowCount = 0;
        //子View的个数
        int childCount = getChildCount();
        //单行子View的宽度
        int rowWidth = 0;
        //子View宽度
        int childWidth;
        //子View高度
        int childHeight;
        //单行中子View的个数
        int childNumInRow = 0;
        //最后一行第一个子View的index
        int tempIndex = 0;
        //除最后一行外,其他单行中子View的总和
        int exceptLastRowNum = 0;
        for (int i = 0; i < childCount; i++) {
            View child = getChildAt(i);
            if (child.getVisibility() == View.GONE) {
                continue;
            }
            CenterLayoutParams params = (CenterLayoutParams) child.getLayoutParams();
            int marginRight = 0, marginTop = 0, marginBottom = 0;
            if (params instanceof MarginLayoutParams) {
                measureChildWithMargins(child, widthMeasureSpec, 0, heightMeasureSpec, measuredHeight);
                marginRight = params.rightMargin;
                marginTop = params.topMargin;
                marginBottom = params.bottomMargin;
            } else {
                measureChild(child, widthMeasureSpec, heightMeasureSpec);
            }
            //子View本身的宽度+子View之间的间距+子View的marginRight值(我这么写偷懒了,mChildSpacing和marginRight不用同时设置值)
            childWidth = child.getMeasuredWidth() + mChildSpacing + marginRight;
            //子View本身的高度+子View的marginTop值+子View的marginBottom+marginTop值(我这么写偷懒了,mRowSpacing和marginBottom + marginTop不用同时设置值)
            childHeight = child.getMeasuredHeight() + mRowSpacing + marginBottom + marginTop;
            //叠加子View的宽度
            rowWidth += childWidth;
            //取出最大的宽度
            maxWidth += Math.max(maxWidth, childWidth);
            //判断是否需要换行
            if (measuredWidth + childWidth > rowLength) {
                //循环后,就可获取,最后一行中第一个子View的index
                tempIndex = i;
                //获取单行的宽度
                rowWidth = rowWidth - childWidth - mChildSpacing - marginRight;
                //存储单行的宽度
                itemLineWidth.add(rowWidth);
                //设置下一行宽度为第一个子View的宽度
                rowWidth = childWidth;
                //行数自增
                ++rowCount;
                //保存测量的宽度
                measuredWidth = childWidth;
                //叠加子View的高度
                maxHeight += childHeight;
                //存储单行中子View的个数
                itemLineNum.add(childNumInRow);
                //叠加获取除了最后一行外,其他行子View的个数的总和
                exceptLastRowNum += childNumInRow;
                //重置子View的个数,因为已经要换行了
                childNumInRow = 1;
            } else {
                //叠加宽度
                measuredWidth += childWidth;
                //叠加单行中的子View个数
                ++childNumInRow;
                //计算出最大的高度
                maxHeight = Math.max(maxHeight, childHeight);
            }

        }
        //最后一行的宽度
        int lastRowWidth = 0;
        int singleHorizalMargin = 0;
        for (int i = tempIndex; i < childCount; i++) {
            View child = getChildAt(i);
            int horizalMargin = 0;
            CenterLayoutParams params = (CenterLayoutParams) child.getLayoutParams();
            if (params instanceof MarginLayoutParams) {
                //获取子View的marginRight
                singleHorizalMargin = horizalMargin = params.rightMargin;
            }
            //叠加最后一行中所有子View的宽度
            lastRowWidth += child.getMeasuredWidth() + mChildSpacing + horizalMargin;
        }
        //获取最后一行子View的个数
        int lastChildCount = childCount - exceptLastRowNum;
        //减去多余的marginRight或mChildSpacing,得到最终的宽度
        lastRowWidth -= mChildSpacing == 0 ? singleHorizalMargin : mChildSpacing;
        //保存最后一行的宽度值
        itemLineWidth.add(lastRowWidth);
        //保存最后一行子View的个数
        itemLineNum.add(lastChildCount);
        //子View的总行数
        mRowTotalCount = rowCount + 1;
        //子View的最大宽度+ViewGroup中的paddingLeft和paddingRight值
        maxWidth = Math.max(maxWidth, getSuggestedMinimumWidth()) + getPaddingRight() + getPaddingLeft();
        //子View的最大高度+ViewGroup中的paddingTop和paddingBottom值
        maxHeight = Math.max(maxHeight, getSuggestedMinimumHeight()) + getPaddingTop() + getPaddingBottom();
        //根据widthMode设置width值
        measuredWidth=widthMode==MeasureSpec.EXACTLY?widthSize:maxWidth;
        //根据heightMode设置height值
        measuredHeight=heightMode==MeasureSpec.EXACTLY?heightSize:maxHeight;
        //设置ViewGroup的宽高
        setMeasuredDimension(resolveSize(measuredWidth, widthMeasureSpec),
                resolveSize(measuredHeight, heightMeasureSpec));
    }

    /**
     * 设置子View的间距
     * @param childSpacing
     */
    public void setChildSpacing(int childSpacing){
        mChildSpacing=childSpacing;
        requestLayout();
    }

    /**
     * 设置子View的marginTop值
     * @param rowSpacing
     */
    public void setRowSpacing(int rowSpacing){
        mRowSpacing=rowSpacing;
        requestLayout();
    }

    @Override
    protected LayoutParams generateLayoutParams(LayoutParams p) {
        return new CenterLayoutParams(p);
    }

    @Override
    public LayoutParams generateLayoutParams(AttributeSet attrs) {
        return new CenterLayoutParams(getContext(), attrs);
    }

    /**
     * @param p
     * @return
     */
    @Override
    protected boolean checkLayoutParams(LayoutParams p) {
        return p instanceof CenterLayoutParams;
    }

    /**
     * 因为需要获取子View的margin值,所以这里需要重写一下该方法
     */
    public static class CenterLayoutParams extends MarginLayoutParams {

        public CenterLayoutParams(MarginLayoutParams source) {
            super(source);
        }

        public CenterLayoutParams(LayoutParams source) {
            super(source);
        }

        public CenterLayoutParams(Context c, AttributeSet attrs) {
            super(c, attrs);
        }

        public CenterLayoutParams(int width, int height) {
            super(width, height);
        }
    }
}

好了,具体思路和写法就这么多了,最后奉上GitHub地址,觉得还行就给个star吧😊。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,332评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,508评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,812评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,607评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,728评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,919评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,071评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,802评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,256评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,576评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,712评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,389评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,032评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,798评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,026评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,473评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,606评论 2 350

推荐阅读更多精彩内容