View 自定义 - 布局 Layout

文章详细介绍了在自定义ViewGroup时如何重写onLayout方法,遍历并计算子元素的位置。onLayout中计算每个子元素的四边距离,并调用子元素的layout方法。同时提到了getMeasuredWidth和getWidth的区别,前者是在measure阶段获取测量后的值,后者在layout之后提供最终尺寸。

一、概念

View 由于没有子元素因此它的 onLayout() 是个空实现。自定义 ViewGroup 时需要重写 onLayout() 方法,遍历并计算每一个子元素四条边到父容器的距离,然后调用子元素的 layout() 传递过去。

override fun onLayout(changed: Boolean, left: Int, top: Int, right: Int, bottom: Int) {...}

形参changed为当前控件大小和位置是否改变了(很耗性能,父容器宽高的改变和子控件请求刷新都会被系统频繁调用),其它为控件自身四边到父容器的距离。

override fun onLayout(changed: Boolean, l: Int, t: Int, r: Int, b: Int) {
    val childLeft = 0
    val childTop = 0
    val childRight = 0
    val childBottom = 0
    //遍历子元素
    for (i in 0 Uuntil childCount) {
        val childView = getChildAt(i)
        //获取测量后的子元素宽高
        val childMeasuredWidth = childView.measuredWidth
        val childMeasuredHeight = childView.measuredHeight
        //计算子元素四边到父容器的距离(结合在onMeasure()中保存的一些全局数据)
        //TODO...
        //确定了子元素在父容器中的位置,调用子元素的 layout() 传递过去
        childView.layout(childLeft, childTop, childRight, childBottom)
    }
}

1.1 getMeasuredXXX() 和 getXXX() 区别

一般情况下是相等的,除了重写 layout() 进行了修改。

getMeasuredWidth()

getMeasuredHeight()

getWidth()

getHeight()

获取测量后的值获取布局后的值(名字中省略了Layouted)
在 measure() 结束后就可以获取到,供后续布局使用。在 layout() 执行后才能获取到,是控件真正的最终尺寸。
通过 setMeasuredDimension( ) 进行设置的。通过视图右边坐标减去左边坐标计算出来的。

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值