對於流式佈局的研究和一點心得

流式佈局現在已經不是什麼新鮮的設計了,經常在各種列表展示文本框的時候可以看到,特別是應用於標籤類的。



其實可以將其看作會自動換行的線性佈局,根據內部的子控件所佔的實際大小重新進行排列。具體實現的思路如下:

 

1.自定義一個控件FlowLayout,繼承ViewGroup。定義水平間距和行與行之間的垂直間距。初始化一個集合lineList,用來存放所有的Line對象(Line的用途後面說)

 

2.重寫onMeasure方法,爲子控件測量寬度,將其歸併分類以便排列布局

 2.1.獲取當前FlowLayout的寬度,注意其實際比較的寬度應減去paddingLefet和paddingRight.

 2.2.創建一個Line對象,這裏爲了方便直接將Line定義在FlowLayout的內部。每一個Line對象就代表了每一行,封裝了存放當前行所有子View的集合,所有子View的寬+水平間距之和,以及所在行的高度。

 2.3.遍歷當前FlowLayout包含的每一個子View,用子View的寬和FlowLayout的實際寬度比較。

 2.4.如果之前創建的Line中沒有子View,則直接存入該View,因爲要保證每行至少有一個子View;

 2.5.如果當前Line的寬+水平間距+子View的寬大於FlowLayout的實際寬度,則該子View需要換行,也就是重新創建一個Line對象存放。注意要先存儲之前的Line對象,否則會缺失View.

 2.6.其它的情況,說明該行所擁有的空間還可以容納下該子View,直接存入

 2.7.如果當前子View是最後一個,那麼需要保存最後的line對象。到此lineList存放了所有的Line,而每個Line又記錄了自己行所有的View.

 2.8.寬度之前已經計算出來,開始計算FlowLayout需要的高度。高度=FlowLayout的paddingTop+paddingBottom+所有行的高度+所有行的行間距。拿計算好的寬高值重新設置當前控件的尺寸setMeasuredDimension(width, height);
  2.9.注意開始的時候清空lineList。因爲onMeasure在View的繪製過程中會多次被回調。如果不清空的話,會進行多次無意義的測量,生成很多空白的子View。
 
3. 重寫onLayout方法,根據分類情況爲子View擺放它們實際應該在的位置
   3.1.遍歷lineList,拿到每一個Line對象.      
    3.2.內循環開始,獲取該Line對象所包含的所有view的集合。
   3.3.獲取該Line對象的留白,也就是FlowLayout的實際寬度除去當前Line所佔的實際寬度值。因爲換行的時候不一定剛好子View的寬度和+間距=FlowLayout的實際寬度。
   3.4.用留白/view集合中的元素個數,得到浮點數值,就是每個子View得到的補償寬度。讓子View重新測量一次,爲其賦值新的寬度。
   3.5.開始擺放。如果子View是每行的第一個,直接擺放。如果不是,參考前一個子View。當前的left是前一個的right+水平間距。right=left+自身寬度
   3.6.注意從第二行開始,每行的top總是比上一行的top多累加一個行高和垂直間距。而行高和行寬不一樣,取內部高度最高的子View的高度值。

4. FlowLayout的實現代碼如下:
public class FlowLayout extends ViewGroup {

    private int horizontalSpacing = 15;//水平間距
    private int verticalSpacing   = 15;//行與行之間的垂直間距

    //用來存放所有的Line對象
    private ArrayList<Line> lineList = new ArrayList<Line>();

    public FlowLayout(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
    }

    public FlowLayout(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    public FlowLayout(Context context) {
        super(context);
    }

    /**
     * 設置水平間距
     *
     * @param horizontalSpacing
     */
    public void setHorizontalSpacing(int horizontalSpacing) {
        this.horizontalSpacing = horizontalSpacing;
    }

    /**
     * 設置垂直間距
     *
     * @param verticalSpacing
     */
    public void setVerticalSpacing(int verticalSpacing) {
        this.verticalSpacing = verticalSpacing;
    }

    /**
     * 分行:遍歷所有的子View,判斷哪幾個子View在同一行(排座位表)
     */
    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {

        lineList.clear();

        //1.獲取FlowLayout的寬度
        int width = MeasureSpec.getSize(widthMeasureSpec);
        //2.獲取用於實際比較的寬度,就是除去2邊的padding的寬度
        int noPaddingWidth = width - getPaddingLeft() - getPaddingRight();

        //3.遍歷所有的子View,拿子View的寬和noPaddingWidth進行比較
        Line line = new Line();//準備Line對象
        for (int i = 0; i < getChildCount(); i++) {
            View childView = getChildAt(i);
            childView.measure(0, 0);//保證能夠獲取到寬高

            //4.如果當前line中沒有子View,則直接存入,因爲要保證每行至少有一個子View;
            if (line.getViewList().size() == 0) {
                line.addLineView(childView);//直接存入
            } else if (line.getLineWidth() + horizontalSpacing + childView.getMeasuredWidth() >
                    noPaddingWidth) {
                //5.如果當前line的寬+水平間距+子View的寬大於noPaddingWidth,則child需要換行
                //要先存放之前的line對象,否則會缺失View
                lineList.add(line);

                line = new Line();//創建新的Line
                line.addLineView(childView);//將當前child放入新的行中
            } else {
                //6.說明當前child應該放入當前Line中
                line.addLineView(childView);
            }

            //7.如果當前child是最後的子View,那麼需要保存最後的line對象
            if (i == (getChildCount() - 1)) {
                lineList.add(line);//保存最後的Line
            }
        }

        //循環結束,lineList存放了所有的Line,而每個Line又記錄了自己行所有的View;
        //現在計算FLowLayout需要的高度
        int height = getPaddingTop() + getPaddingBottom();//先計算上下的padding值
        for (int i = 0; i < lineList.size(); i++) {
            height += lineList.get(i).getLineHeight();//再加上所有行的高度
        }
        height += (lineList.size() - 1) * verticalSpacing;//最後加上所有的行間距

        //設置當前控件的寬高,或者向父View申請寬高
        setMeasuredDimension(width, height);
    }

    /**
     * 去擺放所有的子View,讓每個人真正的坐到自己的位置上
     */
    @Override
    protected void onLayout(boolean changed, int l, int t, int r, int b) {
        int paddingLeft = getPaddingLeft();
        int paddingTop = getPaddingTop();
        for (int i = 0; i < lineList.size(); i++) {
            Line line = lineList.get(i);//獲取Line對象

            //從第二行開始,每行的top總是比上一行的top多一個行高和垂直間距
            if (i > 0) {
                paddingTop += verticalSpacing + lineList.get(i - 1).getLineHeight();
            }

            ArrayList<View> viewList = line.getViewList();//獲取line所包含的所有view的集合

            //1.獲取每行的留白的寬度
            int remainSpacing = getLineRemainSpacing(line);
            //2.計算每個view平均得到的值
            float perSpacing = remainSpacing / viewList.size();

            for (int j = 0; j < viewList.size(); j++) {
                View childView = viewList.get(j);
                //3.將得到的perSpacing增加到view的寬度上面
                int widthSpec = MeasureSpec.makeMeasureSpec((int) (childView.getMeasuredWidth() +
						perSpacing), MeasureSpec.EXACTLY);
                childView.measure(widthSpec, 0);

                if (j == 0) {
                    //如果是每行的第一行,name直接靠左邊擺放
                    childView.layout(paddingLeft, paddingTop, paddingLeft + childView
									.getMeasuredWidth(),
                            paddingTop + childView.getMeasuredHeight());
                } else {
                    //如果不是第一個,需要參考前一個view的right
                    View preView = viewList.get(j - 1);
                    //當前view的left是前一個view的right+水平間距
                    int left = preView.getRight() + horizontalSpacing;
                    childView.layout(left, preView.getTop(), left + childView.getMeasuredWidth(),
							preView.getBottom());
                }
            }
        }
    }

    /**
     *
     * @param   line
     * @return  獲取指定Line的留白
     */
    private int getLineRemainSpacing(Line line) {
        return getMeasuredWidth() - getPaddingLeft() - getPaddingRight() - line.getLineWidth();
    }

    /**
     * 封裝每行的數據,包括所有的子View,行的寬高。每一個Line對象就代表一行
     *
     * @author Administrator
     */
    private class Line {
        private ArrayList<View> viewList;//用來存放當前行所有的子View
        private int             width;//表示所有子View的寬+水平間距
        private int             height;//行的高度

        private Line() {
            viewList = new ArrayList<View>();
        }

        /**
         * 記錄子View
         *
         * @param child
         */
        private void addLineView(View child) {
            if (!viewList.contains(child)) {
                viewList.add(child);

                //1.更新Line的width
                if (viewList.size() == 1) {
                    //說明添加的是第一個子View,那麼line的寬就是子View的寬度
                    width = child.getMeasuredWidth();
                } else {
                    //如果添加的不是第一個子View,那麼應該加等於水平間距和子View的寬度
                    width += child.getMeasuredWidth() + horizontalSpacing;
                }
                //2.更新line的height
                height = Math.max(height, child.getMeasuredHeight());
            }
        }

        /**
         *
         * @return  獲取當前行的寬度
         */
        private int getLineWidth() {
            return width;
        }

        /**
         *
         * @return  當前行的高度
         */
        private int getLineHeight() {
            return height;
        }

        /**
         *
         * @return  獲取當前行的所有的子View的集合
         */
        private ArrayList<View> getViewList() {
            return viewList;
        }
    }

}

 
5. 往FlowLayout裏面填充元素
   5.1.最常見的流式佈局,就是向裏面填充TextView,以此示例    
    5.2.初始化一個字符串集合,一會TextView要顯示的.給FlowLayout設置一個padding值
   5.3.遍歷集合,每次創建一個TextView,可以根據需求設置字體大小和顏色,這裏取固定,但是讓背景色取隨機值。
   5.4.每一次循環時,注意將創建的TextView加入到FlowLayout中去,讓FlowLayout去完成剩下的測量和佈局的事情

示例代碼:
public class MainActivity extends AppCompatActivity {

    FlowLayout mFlowLayout;

    String[] appNames = {"Java", "Android", "C", "C++", "JavaScript", "Python", "Ruby", "C#",
            "Node.js", "HTML", "PHP", "COCOS2D-X"};

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        mFlowLayout = findViewById(R.id.fl_container);

        int layoutPadding = Utils.getDimens(this, R.dimen.dp10);
        mFlowLayout.setPadding(layoutPadding, layoutPadding, layoutPadding, layoutPadding);

        for (int i = 0; i < appNames.length; i++) {
            final TextView tv = new TextView(this);
            tv.setText(appNames[i]);

            GradientDrawable gradientDrawable = new GradientDrawable();
            gradientDrawable.setShape(GradientDrawable.RECTANGLE);
            int dp5 = Utils.getDimens(this, R.dimen.dp5);
            gradientDrawable.setCornerRadius(dp5);
            gradientDrawable.setColor(Color.rgb(Utils.createRandomColor(), Utils
                    .createRandomColor(), Utils.createRandomColor()));

            GradientDrawable gradientDrawable2 = new GradientDrawable();
            gradientDrawable2.setShape(GradientDrawable.RECTANGLE);
            gradientDrawable2.setCornerRadius(dp5);
            gradientDrawable2.setColor(Color.rgb(Utils.createRandomColor(), Utils
                    .createRandomColor(), Utils.createRandomColor()));

            tv.setTextColor(Color.WHITE);
            tv.setTextSize(16);
            tv.setGravity(Gravity.CENTER);
            int paddingValue = Utils.getDimens(this,R.dimen.dp5);
            tv.setPadding(paddingValue,paddingValue,paddingValue,paddingValue);

            StateListDrawable stateListDrawable = new StateListDrawable();
            stateListDrawable.addState(new int[]{android.R.attr.state_pressed},
                    gradientDrawable);
            stateListDrawable.addState(new int[]{}, gradientDrawable2);

            tv.setBackground(stateListDrawable);
            tv.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {

                }
            });

            mFlowLayout.addView(tv);
        }

    }
}



發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章