Android高级UI开发(三十六)measure测量实例

举报
yd_57386892 发表于 2020/12/28 22:44:30 2020/12/28
【摘要】           今天我们来通过一个实例来讲解一下view的测量绘制过程。我们打算做一个瀑布流标签,就是各个标签的宽高都不一样,当一行占满时就自动换行,示意图如下: ,这个与Grid是有区别的,Grid是每行有固定的列数,而我们这个瀑布流标签是一行占满为止才换行。我们打算做一个自定义容器,这个容器内部的各个子控件会自动按瀑布流标签的形式摆放,对各个子控件的类型没有要求...

          今天我们来通过一个实例来讲解一下view的测量绘制过程。我们打算做一个瀑布流标签,就是各个标签的宽高都不一样,当一行占满时就自动换行,示意图如下:

,这个与Grid是有区别的,Grid是每行有固定的列数,而我们这个瀑布流标签是一行占满为止才换行。我们打算做一个自定义容器,这个容器内部的各个子控件会自动按瀑布流标签的形式摆放,对各个子控件的类型没有要求。这样的话,我们创建一个类FlowLayout,让它继承ViewGroup,然后重写onMeasure函数,onLayout函数,至于为何不重写onDraw函数,我们实现了功能后再作分析。

 

接下来我们先分析FlowLayout,它的源代码如下:


      package com.test.measure;
      import android.content.Context;
      import android.util.AttributeSet;
      import android.util.DisplayMetrics;
      import android.util.Log;
      import android.view.View;
      import android.view.ViewGroup;
      import java.util.ArrayList;
      import java.util.List;
      /**
       * Created by xiaowei on 2019/12/5.
       */
      public class FlowLayout extends ViewGroup {
      private static final String TAG = "FlowLayout" ;
      /**
       * 用来保存每行views的列表
       */
      private List<List<View>> mViewLinesList = new ArrayList<>();
      /**
       * 用来保存行高的列表
       */
      private List<Integer> mLineHeights = new ArrayList<>();
      public FlowLayout(Context context, AttributeSet attrs) {
      super(context, attrs);
       }
      @Override
      public LayoutParams generateLayoutParams(AttributeSet attrs) {
      return new MarginLayoutParams(getContext(),attrs);
       }
      @Override
      protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
      super.onMeasure(widthMeasureSpec, heightMeasureSpec);
       mViewLinesList.clear();
       mLineHeights.clear();
      // 获取父容器为FlowLayout设置的测量模式和大小
       int iWidthMode = MeasureSpec.getMode(widthMeasureSpec); //EXACT
       int iHeightMode = MeasureSpec.getMode(heightMeasureSpec); //AT_MOST
       int iWidthSpecSize = MeasureSpec.getSize(widthMeasureSpec);   //1080,EXACT
       int iHeightSpecSize = MeasureSpec.getSize(heightMeasureSpec);  //1950(有底部虚拟导航),2076(虚拟导航不可见),AT_MOST
       int measuredWith = 0;  //这个自定义容器的测量宽度
       int measuredHeight = 0;//这个自定义容器的测量高度
       int iCurLineW = 0; //当前行宽,不断累加,超过屏幕宽度,则转向第二行
       int iCurLineH = 0; //当前行高
      if(iWidthMode == MeasureSpec.AT_MOST )
       {
       Log.i(TAG,"widthMode=AT_MOST,iWidthSpecSize="+iWidthSpecSize);
       }else if(iWidthMode == MeasureSpec.UNSPECIFIED)
       {
       Log.i(TAG,"widthMmode=UNSPECIFIED,iWidthSpecSize="+iWidthSpecSize);
       }else if(iWidthMode == MeasureSpec.EXACTLY)
       { Log.i(TAG,"widthMode=EXACTLY,iWidthSpecSize="+iWidthSpecSize);
       }
      if(iHeightMode == MeasureSpec.AT_MOST )
       {
       Log.i(TAG,"heightMode=AT_MOST,heightSize="+iHeightSpecSize);
       }else if(iHeightMode == MeasureSpec.UNSPECIFIED)
       {
       Log.i(TAG,"heightMode=UNSPECIFIED,heightSize="+iHeightSpecSize);
       }else if(iHeightMode == MeasureSpec.EXACTLY)
       {
       Log.i(TAG,"heightMode=EXACTLY,heightSize="+iHeightSpecSize);
       }
      if(iWidthMode == MeasureSpec.EXACTLY && iHeightMode == MeasureSpec.EXACTLY){
       measuredWith = iWidthSpecSize;
       measuredHeight = iHeightSpecSize;
       }else{
       int iChildWidth;
       int iChildHeight;
       int childCount = getChildCount();
       List<View> viewList = new ArrayList<>();
      for(int i = 0 ; i < childCount ; i++){
       View childView = getChildAt(i);
       measureChild(childView, widthMeasureSpec,heightMeasureSpec);
       MarginLayoutParams layoutParams = (MarginLayoutParams) childView.getLayoutParams();
       iChildWidth = childView.getMeasuredWidth() + layoutParams.leftMargin +
       layoutParams.rightMargin;
       iChildHeight = childView.getMeasuredHeight() + layoutParams.topMargin +
       layoutParams.bottomMargin;
      if(iCurLineW + iChildWidth > iWidthSpecSize){
      //将要折行了,折行之前先记录当前行的宽度,如果当前行更宽一些,那么就把当前行的宽度给measuredWith(作为FlowLayout的宽度),iWidthSpecSize为FlowLayout的父容器LineareLayout给FlowLayout指定的精确宽度(EXACT模式),屏幕宽度1080
      /**1、记录当前行的信息***/
      //1、记录当前行的最大宽度,高度累加
       measuredWith = Math.max(measuredWith,iCurLineW);
       measuredHeight += iCurLineH;
      //2、将当前行的viewList添加至总的mViewsList,将行高添加至总的行高List
       mViewLinesList.add(viewList);//将当前行的控件组添加到“行二维数组”中
       mLineHeights.add(iCurLineH); //记录每一行的高度
      /**2、记录新一行的信息***/
      //1、重新赋值新一行的宽、高
       iCurLineW = iChildWidth;
       iCurLineH = iChildHeight;
      // 2、新建一行的viewlist,添加新一行的view
       viewList = new ArrayList<View>();
       viewList.add(childView);
       }else{
      // 记录某行内的消息
      //1、行内宽度的叠加、高度比较
       iCurLineW += iChildWidth;
       iCurLineH = Math.max(iCurLineH, iChildHeight);
      // 2、添加至当前行的viewList中
       viewList.add(childView);  //记录某一行的子控件,将来是二维数组的列
       }
      /*****3、为最后一行 换行,否则measuredHeight就会少加最后一行,并且mViewLinesList也就少加最后一行控件**********/
      if(i == childCount - 1){
      //1、记录当前行的最大宽度,高度累加
       measuredWith = Math.max(measuredWith,iCurLineW);
       measuredHeight += iCurLineH;
      //2、将当前行的viewList添加至总的mViewsList,将行高添加至总的行高List
       mViewLinesList.add(viewList);
       mLineHeights.add(iCurLineH);
       }
       }
       }
      // 最终目的
       setMeasuredDimension(measuredWith,measuredHeight);//实际ViewGroup的宽高:measuredWith=1072,measuredHeight=397
       }
      @Override
      protected void onLayout(boolean changed, int l, int t, int r, int b) {
       int left,top,right,bottom;
       int curTop = 0;
       int curLeft = 0;
       int lineCount = mViewLinesList.size();
      for(int i = 0 ; i < lineCount ; i++) {
       List<View> viewList = mViewLinesList.get(i);
       int lineViewSize = viewList.size();
      for(int j = 0; j < lineViewSize; j++){
       View childView = viewList.get(j);
       MarginLayoutParams layoutParams = (MarginLayoutParams) childView.getLayoutParams();
       left = curLeft + layoutParams.leftMargin;
       top = curTop + layoutParams.topMargin;
       right = left + childView.getMeasuredWidth();
       bottom = top + childView.getMeasuredHeight();
       childView.layout(left,top,right,bottom);
       curLeft += childView.getMeasuredWidth() + layoutParams.leftMargin + layoutParams.rightMargin;
       }
       curLeft = 0;
       curTop += mLineHeights.get(i);
       }
       mViewLinesList.clear();
       mLineHeights.clear();
       }
      }
  
 

1. widthMeasureSpec与heightMeasureSpec的测量模式与大小


        我们首先看一下onMeasure函数,它的两个参数widthMeasureSpec和heightMeasureSpec,这2个参数是FlowLayout的父容器即Xml里的LinearLayout给FlowLayout的宽高规格。我们专门下了一段代码来打印日志,输出widthMeasureSpec的模式和大小,以及heightMeasureSpec的测量模式和大小。 

1.1  猜想测量模式与大小  

            我们可否根据XML布局文件,先猜测一下它们的测量模式和大小,xml布局如下:我们看到LinearLayout的  android:layout_width="match_parent",同时FlowLayout的android:layout_width="match_parent",LinearLayout的match_parent就是屏幕的宽度,这里是1080px,同时FlowLayout的layout_width匹配到LinearLayout的match_parent,也就是说FlowLayout的宽度和LinearLayout的宽度一样,那这样明显就是一个精确值1080,即LinearLayout给FlowLayout的测量宽度规格widthMeasureSpec,它的测量模式一定是EXACT,size是1080px。

            接下来我们看一下高度规格,LinearLayout的layout_height高度为match_parent,即整个屏幕的可用高度,这里是1950,本来应该是2000多像素,可能是把状态栏与虚拟导航栏除外了吧。而FlowLayout的layout_height为wrap_content,高度随它自己的内容高度而改变,这样的话,父容器LinearLayout有一个固定的高度match_parent(1950),子控件FlowLayout的内容又不是确定的,那就猜想子控件FlowLayout的高度至多不超过父容器LinearLayout的高度,即LinearLayout给的heightMeasureSpec高度规格的测量模式是AT_MOST, 同时size就是1950,这个规格意味着子控件FlowLayout的高度是动态变化的且最大不能超过1950px。

1.2  猜想与实际测量模式与大小的对比

我们对比一下打印出的日志,发现与我们猜想的一样,日志如下:


      FlowLayout: widthMode=EXACTLY,iWidthSpecSize=1080
      FlowLayout: heightMode=AT_MOST,heightSize=1876
  
 

细心的读者会发现size和我们上述1.1中的不一样,这是因为我换了一台手机,呵呵。

2. onMeasure函数深入浅出

好,接下来我们继续分析onMeasure代码,
     if(iWidthMode == MeasureSpec.EXACTLY && iHeightMode == MeasureSpec.EXACTLY){
            measuredWith = iWidthSpecSize;
            measuredHeight = iHeightSpecSize;
        }
显然我们现在的测量规格不会进入这个分支,这段代码表示如果父容器给当前FlowLayout宽、高测量模式都是精确值,那我们就不用再测量了,FlowLayout的宽高直接就是widthMeasureSpec和heightMeasureSpec中给定的大小,在这里再说一下测量规格是一个32位整数,高2位表示测试模式,低30位表示给定的大小值。那我们这里会进入for循环这段代码:
for(int i = 0 ; i < childCount ; i++){
......
}

在这个for循环里测量每一个子view(FlowLayout内的各个textview标签控件)的大小(宽,高),因为我们要知道当前父容器FlowLayout的测量大小,就得先知道各个子控件的宽高。FlowLayout的最终宽度肯定是各行控件中的最大行宽度。高度是多行控件累加的总高度,而每一行的高度应该取该行中高度最高的childView. 在这里我们就有一个疑问,我们FlowLayout的onMeasure传递进来的widthMeasureSpec规格不是已经是EXACT模式了吗?而且大小也给定了,即FlowLayout的宽度就是widthMeasureSpec中的大小1080px,那我们为什么还要根据一行child的宽度来再算一次FlowLayout的宽度呢,我现在只能说这样更精确一些,算下来FlowLayout的最终测量宽度是1072px.和widthMeasureSpec规格中给定的size差不多。那么如何来计算呢?

2. 1 先测量childView

第一步先测量childview


       View childView = getChildAt(i);
       measureChild(childView, widthMeasureSpec,heightMeasureSpec);
       MarginLayoutParams layoutParams = (MarginLayoutParams) childView.getLayoutParams();
       iChildWidth = childView.getMeasuredWidth() + layoutParams.leftMargin +
       layoutParams.rightMargin;
       iChildHeight = childView.getMeasuredHeight() + layoutParams.topMargin +
       layoutParams.bottomMargin;
  
 


这段代码中的measureChild函数测量出了子childview的可视宽、高(不包括margin,也就是只包含内容+padding),iChildWidth 和iChildHeight 是将childview的margin属性考虑进去了,我们容器FlowLayout要容纳这些childview肯定要给这些margin留有空间。

2.2 计算出FlowLayout的宽、高

第二步,计算出行中的最大行宽,与各行的累加高度。

   if(iCurLineW + iChildWidth > iWidthSpecSize){
                    //将要折行了,折行之前先记录当前行的宽度,如果当前行更宽一些,那么就把当前行的宽度给measuredWith(作为FlowLayout的宽度),iWidthSpecSize为FlowLayout的父容器LineareLayout给FlowLayout指定的精确宽度(EXACT模式),屏幕宽度1080


                    /**1、记录当前行的信息***/

                    //1、记录当前行的最大宽度,高度累加
                    measuredWith = Math.max(measuredWith,iCurLineW);
                    measuredHeight += iCurLineH;
                    //2、将当前行的viewList添加至总的mViewsList,将行高添加至总的行高List
                    mViewLinesList.add(viewList);//将当前行的控件组添加到“行二维数组”中
                    mLineHeights.add(iCurLineH); //记录每一行的高度

                    /**2、记录新一行的信息***/

                    //1、重新赋值新一行的宽、高
                    iCurLineW = iChildWidth;
                    iCurLineH = iChildHeight;

                    // 2、新建一行的viewlist,添加新一行的view
                    viewList = new ArrayList<View>();
                    viewList.add(childView);

                }else{
                    // 记录某行内的消息
                    //1、行内宽度的叠加、高度比较
                    iCurLineW += iChildWidth;
                    iCurLineH = Math.max(iCurLineH, iChildHeight);

                    // 2、添加至当前行的viewList中
                    viewList.add(childView);  //记录某一行的子控件,将来是二维数组的列
                }

iCurLine0W 用于记录每一行的宽度,初始值为0,假设刚开始摆了一个Textview,iCurLine0W 的值就会变成iCurLineW +=iChildWidth; 也就是先进入else这个分子,iCurLineH 就是iChildHeight。当摆放第二个Textview,的时候iCurLine0W 继续累加第二个Textview的iChildWidth:iCurLine0W +=iChildWidth; 然后iCurLineH会和第二个 Textview的iChildHeight)对比,取最大值,总之当前行高iCurLineH的值总以这行中高度最高的Textview为准。然后,继续摆第三个Textview,假设这时的行宽累加:iCurLineW += iChildWidth将要超过FlowLayout的宽度iWidthSpecSize,那么就要换行显示,即进入if分支。换行前,先把这第一行的行宽iCurLineW作为measuredWith(FlowLayout的宽):  measuredWith = Math.max(measuredWith,iCurLineW); 这里会和后续行的行宽比较大小,总之measuredWith取所有行中最宽的一个值。同时高度measuredHeight也要累加一行的高度iCurLineH,即:measuredHeight += iCurLineH。FlowLayout的测量高度measuredHeight最终是各行高度之和。   mViewLinesList.add(viewList);//将当前行的控件组添加到“行二维数组”中
                    mLineHeights.add(iCurLineH); //记录每一行的高度。
这两行代码mViewLinesList二维数组,是为了保存每一行的子控件,包含多行。mLineHeights一维数组是保存每一行的高度。在这里保存是为了onLayout中摆放方便,在onLayout中会直接摆放mViewLinesList已归类的每一行控件,也就是说在onLayout中就没必要判断什么时候该折行了,因为每一行包含多少个控件,哪些控件该折行到下一行,mViewLinesList都已记录了。为onLayout省去了大量的代码,到时候直接摆放mViewLinesList里面的控件就行。

   //1、重新赋值新一行的宽、高
                    iCurLineW = iChildWidth;
                    iCurLineH = iChildHeight;

                    // 2、新建一行的viewlist,添加新一行的view
                    viewList = new ArrayList<View>();
                    viewList.add(childView);
这就是刚才上一行每有空间容纳的那个控件,它应该放在下一行,这时iCurLineW 下一行宽的初始值就是这个控件的宽。下一行的iCurLineH行高初始值 就是iChildHeight; 同时下一行控件数组的第一个元素就是当前childView.

    /*****3、为最后一行 换行,否则measuredHeight就会少加最后一行,并且mViewLinesList也就少加最后一行控件**********/
                if(i == childCount - 1){
                    //1、记录当前行的最大宽度,高度累加
                    measuredWith = Math.max(measuredWith,iCurLineW);
                    measuredHeight += iCurLineH;

                    //2、将当前行的viewList添加至总的mViewsList,将行高添加至总的行高List
                    mViewLinesList.add(viewList);
                    mLineHeights.add(iCurLineH);

                }

 if(i == childCount - 1)判断是最后一个孩子的话,强制触发换行,因为最后一行,比如只放了一个TextView的时候,它的宽度不足以触发上述if分支,也就是说measuredWith和measuredHeight就少计算一次,至少高度的累加上会少加这最后一行的高度:  measuredHeight += iCurLineH;

最终通过测量了所有child的大小,以及递归得到了FlowGroup的宽高(measuredWith,measuredHeight);
以下这行代码,用于将测量出的FlowGroup的宽高确定下来。将来绘制出来的FlowGroup占用的空间也就是这么大。

2.3  保存FlowLayout的大小


  setMeasuredDimension(measuredWith,measuredHeight);

3. onLayout摆放分析

核心代码如下:
    for(int i = 0 ; i < lineCount ; i++) {
            List<View> viewList = mViewLinesList.get(i);
            int lineViewSize = viewList.size();
            for(int j = 0; j < lineViewSize; j++){
                View childView = viewList.get(j);
                MarginLayoutParams layoutParams = (MarginLayoutParams) childView.getLayoutParams();

                left = curLeft + layoutParams.leftMargin;
                top = curTop + layoutParams.topMargin;
                right = left + childView.getMeasuredWidth();
                bottom = top + childView.getMeasuredHeight();
                childView.layout(left,top,right,bottom);
                curLeft += childView.getMeasuredWidth() + layoutParams.leftMargin + layoutParams.rightMargin;
            }
            curLeft = 0;
            curTop += mLineHeights.get(i);
        }

也就是for循环每一行childview,通过mViewLinesList.get(i)获得每一行的childview,然后根据每一个childview的宽、高、margin来 确定 childview的摆放位置,left,就是childview要摆放的左边缘,它肯定在leftMargin的右边, top,也一样考虑topMargin,childview的上边缘要在topMargin下。right,右边缘不需要考虑rightMarin。bottom,下边缘同样不用考虑bottomMargin .
最后,调用childView.layout(left,top,right,bottom)函数,传入left,top,right,bottom四个参数,实质就是一个矩形的左上角和右下角2个点的坐标,最终摆放了这个childView. 一行中第一个控件摆放完后,curLeft左边缘就要右移一个childview的宽(包含leftMargin和rightMargin),为摆放一行中的下一个childView做准备。当嵌套的for循环执行完后,就到了下一行的摆放,这是curLeft初始为0重新开始横向摆放,  curTop 就要下移一行的高度:  curTop += mLineHeights.get(i);

备注:  mViewLinesList.clear();
        mLineHeights.clear();  无论是onMeasure中的还是onLaout中的clear()都是为了避免重复调用onMeasure和onLaout导致数组内容不断重复增加的BUG.

好了,至此测量和摆放过程已分析完毕,这时候可以运行代码查看效果了。今天先到这里,没有鼠标,每一个字都很难敲上去。下一次我们来分析最后一个问题,为什么我们没有重写onDraw函数,也绘制出了这个瀑布流标签。

 

源码地址:https://download.csdn.net/download/gaoxiaoweiandy/12008467

文章来源: blog.csdn.net,作者:冉航--小虾米,版权归原作者所有,如需转载,请联系作者。

原文链接:blog.csdn.net/gaoxiaoweiandy/article/details/103315173

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。