lmw
2025-04-14 1c1851f2b877adf8a5ae420f8f5adb1e5f2f9771
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
package com.sinata.xqmuse.views;
 
import android.content.Context;
import android.content.res.TypedArray;
import android.util.AttributeSet;
import android.view.View;
import android.view.ViewGroup;
 
import com.sinata.xqmuse.R;
 
 
/**
 * Created by lmw on 2016/11/8.
 */
 
public class FlowLayout extends ViewGroup{
    private float mVerticalSpacing; //每个item纵向间距
    private float mHorizontalSpacing; //每个item横向间距
    public FlowLayout(Context context) {
        this(context,null);
    }
 
    public FlowLayout(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        inial(context,attrs);
    }
 
    public FlowLayout(Context context, AttributeSet attrs) {
        this(context, attrs,0);
        inial(context,attrs);
    }
 
    public void setHorizontalSpacing(float pixelSize) {
        mHorizontalSpacing = pixelSize;
    }
    public void setVerticalSpacing(float pixelSize) {
        mVerticalSpacing = pixelSize;
    }
 
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        //获取容器设置的padding
        int mPaddingLeft = getPaddingLeft();
        int mPaddingRight = getPaddingRight();
        int mPaddingTop = getPaddingTop();
        int mPaddingBottom = getPaddingBottom();
        //获取布局要求给的宽度大小
        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        //获取布局要求给的 高度的mode
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);
        //获取布局要求给的 高度大小
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);
 
        //容器存放所有子View所需的总高度
        int totalHeight = mPaddingTop + mPaddingBottom;
        //某行已使用的宽度
        int lineUsedSize = mPaddingLeft + mPaddingRight;
        //用来保存某行最后占的高度,需要取同一行中最高的子View的高度
        int lineHeight =  0;
 
        //循环遍历每一个子View
        for (int i = 0; i < getChildCount(); i++) {
            View child = getChildAt(i);
            if(child.getVisibility() == GONE){
                //如果子View被设置为隐藏,就跳过这个子View
                continue;
            }
            //让子View自己去测量自己,测量之后就可以调用子View的getMeasureWidth方法获取到测量宽
            measureChild(child,widthMeasureSpec,heightMeasureSpec);
            int childWidth =  child.getMeasuredWidth(); //获取子View的测量宽
            int childHeight = child.getMeasuredHeight(); //获取子View的测量高
            //获取子View的布局参数LayoutParams
            MarginLayoutParams mlps = (MarginLayoutParams) child.getLayoutParams();
            //计算这个孩子一共需要多少空间
            int spaceWidth = childWidth + mlps.leftMargin + mlps.rightMargin;
            int spaceHeight = childHeight + mlps.topMargin + mlps.bottomMargin;
 
            if(lineUsedSize + spaceWidth <= widthSize){
                lineUsedSize += spaceWidth + mHorizontalSpacing;
                //只要子View的高度大与之前保存的行高,就将子View的高度赋值给行高
                if(spaceHeight > lineHeight){
                    lineHeight = spaceHeight;
                }
            }else{ //这个子View不够摆放,将放到下一行
                totalHeight += lineHeight + mVerticalSpacing;
                lineUsedSize = mPaddingLeft + mPaddingRight + spaceWidth;
                lineHeight = spaceHeight;
            }
        }
        if(heightMode == MeasureSpec.EXACTLY){
            totalHeight = heightSize;
        }else{
            totalHeight += lineHeight;
        }
        setMeasuredDimension(widthSize,totalHeight);
    }
 
    /**
     * 给子控件布局,也就是设置子控件在容器中的位置
     * @param changed
     * @param l
     * @param t
     * @param r
     * @param b
     */
    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        //获取给容器设置的padding
        int mPaddingLeft = getPaddingLeft();
        int mPaddingRight = getPaddingRight();
        int mPaddingTop = getPaddingTop();
        //子View
        int lineX = mPaddingLeft;
        int lineY = mPaddingTop;
        int lineWidth = r - l;
        //某行已使用的宽度
        int lineUsed = mPaddingLeft + mPaddingRight;
        int lineHeight = 0;
        for (int i = 0; i < this.getChildCount(); i++) {
            View child = this.getChildAt(i);
            if (child.getVisibility() == GONE) {
                continue;
            }
            int childWidth =  child.getMeasuredWidth(); //获取子View的测量宽
            int childHeight = child.getMeasuredHeight(); //获取子View的测量高
            //获取子View的布局参数LayoutParams
            MarginLayoutParams mlps = (MarginLayoutParams) child.getLayoutParams();
            //计算这个孩子一共需要多少空间
            int spaceWidth = childWidth + mlps.leftMargin + mlps.rightMargin;
            int spaceHeight = childHeight + mlps.topMargin + mlps.bottomMargin;
 
            if(lineUsed + spaceWidth > lineWidth){ //不够放了
                lineY += lineHeight + mVerticalSpacing;
                //变量的重新设置
                lineUsed = mPaddingLeft + mPaddingRight;
                lineX = mPaddingLeft;
                lineHeight = 0;
            }
            //设置子View在容器中的位置
            child.layout(lineX+mlps.leftMargin,lineY+mlps.topMargin,lineX+mlps.leftMargin+childWidth,lineY+mlps.topMargin+childHeight);
            if(spaceHeight > lineHeight){
                lineHeight = spaceHeight;
            }
            lineUsed += spaceWidth + mHorizontalSpacing;
            lineX += spaceWidth + mHorizontalSpacing;
        }
    }
 
    @Override
    protected LayoutParams generateLayoutParams(LayoutParams p) {
        return new MarginLayoutParams(p);
    }
 
    @Override
    public LayoutParams generateLayoutParams(AttributeSet attrs)
    {
        return new MarginLayoutParams(getContext(), attrs);
    }
 
    @Override
    protected LayoutParams generateDefaultLayoutParams() {
        return new MarginLayoutParams(super.generateDefaultLayoutParams());
    }
 
    /**
     * 初始化自定义属性
     * @param context
     * @param attrs
     */
    private void inial(Context context,AttributeSet attrs)  {
        TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.FlowLayout);
        float horizontal = typedArray.getDimension(R.styleable.FlowLayout_HorizontalSpacing,15);
        float verticel = typedArray.getDimension(R.styleable.FlowLayout_VerticalSpacing,15);
        setHorizontalSpacing(horizontal);
        setVerticalSpacing(verticel);
        typedArray.recycle();
 
    }
}