一起Talk Android吧(第四百六十七回:实现自定义ViewGroup中的测量功能)

文章介绍了自定义ViewGroup中的测量功能,强调了在ViewGroup中除了测量自身,还需测量其子View。通过重写onMeasure()方法,根据不同的测量模式计算ViewGroup及其子View的宽度和高度。示例代码展示了如何遍历并计算子View尺寸,以及依据测量模式设置最终的尺寸。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


各位看官们大家好,上一回中咱们说的例子是"实现自定义View中的测量功能",这一回中咱们说的例子是" 实现自定义ViewGroup中的测量功能"。闲话休提,言归正转,让我们一起Talk Android吧!

功能介绍

测量功能是自定义View中的主要功能,测量主要是测量View的宽度和高度,只需要重写View类的onMeasure()方法就可以实现测量功能。测量完成后我们就可以获取到
View的的宽度和高度了。注意:它测量出的宽度和高度值需要使用getMeasuredWidth()方法获取,而且是在onMeasure()方法之后。

此功能对于自定义ViewgGroup也同样适用,毕竟ViewGroup也是一种view。既然一样,那么为会么还要去介绍呢?因为自定义viewgGroup时除了需要测量ViewGroup自身外还需要测量它的子view.

重写方法

重写方法时需要依据不同的测量标准来计算View的宽度和长度,测量的标准有三个,详细内容可以看我之前的博客,我们在这里只介绍重写方法的思路。这个思路分四步:

  1. 获取测量标准和父类的宽度和高度;
  2. 接着依据不同的测量模式计算当前ViewGroup的宽度和高度;
  3. 测量ViewGroup中子View的高度与宽度并且计算出当前ViewGroup的高度和宽度;
  4. 把计算好的宽度和高度设置到当前ViewGroup上;

上面步骤中提到的测量模式与宽度和高度的计算方法遵循以下规则:
如果使用了EXACTLY测量模式,那么就使用父View的测量值。如果使用了EXACTLY以外的测量模式,那么就使用步骤3中计算出的宽度和高度。这个EXACTLY测量模式
对应我们宽度和高度属性中的wrap_conten和xxdp这两种属性值。

该思路比View重写测量方法时多了一步,这一步主要是计算子View的宽度和高度,接下来我介绍如何计算子View的宽度或者高度:

  • 首先遍历当前ViewGroup的所有子view,并且计算出子view的宽度或者高度;
  • 然后把所有子view的宽度或者高度累加起来就是当前ViewGroup的宽度或者高度;

这里需要注意,如果当前ViewGroup的方向是垂直方向,那么所有子view高度累加起来就是ViewGroup的高度,而ViewGrop的宽度是所有子View中最宽的值。如果当前ViewGroup的方向是水平方向,那么所有子view宽度累加起来就是ViewGroup的宽度,而ViewGrop的高度是所有子View中最高的值。

示例代码

介绍完如何重写onMeasure()方法后,我们通过代码来演示具体的实现过程:

//获取父控件原来的长宽值和测量模式
int heightMeasureMode = MeasureSpec.getMode(heightMeasureSpec);
int widthMeasureMode = MeasureSpec.getMode(widthMeasureSpec);
int fatherHeight  = MeasureSpec.getMode(heightMeasureSpec);
int fatherWidth = MeasureSpec.getMode(widthMeasureSpec);

//通过子控件的长宽计算自已的长宽
int width, height;
for(i=0; i<getChildCount;i++) {
    View child = getChildAt(i);
    measureChild(view,widthMeasureSpec,heightMeasureSpec);
    int childHeight = child.getMeasureHeight;
    int childWidth = child.getMeasureWidth;
    if(垂直布局)  {
        height += childHeight;
        width = Math.max(width,childWidth);
    }else {
        width += childWidth;
        height = Math.max(height,childHeight);
    }
}


//依据不同的测量模式,使用不同的计算方法
swith(widthMeasureMode) //使用MeasureSpec.getMode(heightMeasureSpec)获取来
{
    case MeasureSpec.EXACTLY: //match_parent or xxdp
    fatherWidth = fatherWidth;
    break;
    case MeasureSpec.AT_MOST: //wrap_content
    fatherWidth = width;
    break;
    case MeasureSpec.UNSPECIFIED; //末指定
    fatherWidth = width;
    break;
    default:
    fatherWidth = width;
    break;
}
//height计算方法同上,不再列出
//设置长度,这一步很重要
    setMeasuredDimension(width,height);

看官们,关于"实现自定义ViewGroup中的测量功能"的例子咱们就介绍到这里,欲知后面还有什么例子,且听下回分解!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

talk_8

真诚赞赏,手有余香

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值