对MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST)的一点理解

本文介绍了一种解决ScrollView中嵌套ListView导致显示不完全的方法。通过自定义ListView并重写onMeasure方法,设置MeasureSpec使ListView能完全展示其内容。

之前 遇到ScrollView中嵌入ListView,GridView冲突的解决(让ListView全显示出来) 链接
网上查找资料,代码大致如下:

import android.content.Context;
import android.util.AttributeSet;
import android.widget.ListView;

public class ListViewMeasur extends ListView {

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

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

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

	@Override
	public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
//
		int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,
				MeasureSpec.AT_MOST);
		super.onMeasure(widthMeasureSpec, expandSpec);
	}
}

其中对这段代码有疑惑:

	@Override
	public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
//
		int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,
				MeasureSpec.AT_MOST);
		super.onMeasure(widthMeasureSpec, expandSpec);
	}

网上也有解释的文章 比如链接 文章对 测量模式模式的讲解十分详细,但是本文通过源码来进一步理解。

int expandSpec = MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2,MeasureSpec.AT_MOST); 

作为参数onMeasure()方法,很明显作为代表测量的高度

super.onMeasure(widthMeasureSpec, expandSpec);

那接下来通过源码看一下调用的suMeasureSpec.makeMeasureSpec()方法。

    public static int makeMeasureSpec(int size, int mode) {
            if (sUseBrokenMakeMeasureSpec) {
                return size + mode;
            } else {
                return (size & ~MODE_MASK) | (mode & MODE_MASK);
            }
        }

MeasureSpec是一个32位的int值,其中高2位代表测量模式,低30位代表测量大小。
在makeMeasureSpec中看大 return size + mode; 现在通过一段代码模拟一下这个过程。
模拟makeMeasureSpec 输出的是10011111111111111111111111111111
最高两位是10的时候表示"最大模式"。即MeasureSpec.AT_MOST

public class URShift {
	private static final int MODE_SHIFT = 30;
	public static final int AT_MOST     = 2 << MODE_SHIFT;
	public static void main(String[] args) {
		int i =Integer.MAX_VALUE ;
		System.out.println(Integer.toBinaryString(i));
		//~1111111111111111111111111111111
		
		System.out.println(Integer.toBinaryString(makeMeasureSpec(Integer.MAX_VALUE >> 2,
				AT_MOST)));
     	//~10011111111111111111111111111111
	}
	  public static int makeMeasureSpec(int size, int mode) {
		  return size + mode;
      }
}

其实MeasureSpec.makeMeasureSpec(Integer.MAX_VALUE >> 2, MeasureSpec.AT_MOST)就是生成一个符合MeasureSpec的一个32位的包含测量模式和测量高度的int值。

我是IT小王,如果喜欢我的文章,可以扫码关注我的微信公众号
在这里插入图片描述

private fun createViewGroupSkeleton(group: ViewGroup, config: SkeletonConfig): FrameLayout { val context = group.context val skeletonContainer = FrameLayout(context) // 测量整个 group 以确保 layout 已完成 if (group.width == 0 || group.height == 0) { group.measure( View.MeasureSpec.makeMeasureSpec(group.layoutParams.width, View.MeasureSpec.EXACTLY), View.MeasureSpec.makeMeasureSpec(group.layoutParams.height, View.MeasureSpec.EXACTLY) ) group.layout(group.left, group.top, group.right, group.bottom) } // 获取父容器在窗口中的位置(用于坐标转换) val parentLocation = IntArray(2) group.getLocationInWindow(parentLocation) val parentX = parentLocation[0] val parentY = parentLocation[1] for (i in 0 until group.childCount) { val child = group.getChildAt(i) Log.d("SkeletonFactory", "ViewGroup type: ${group.javaClass.simpleName}, id: ${if (group.id == View.NO_ID) "NO_ID" else group.resources.getResourceEntryName(group.id)}, childCount: ${group.childCount}") // 跳过占位视图如 Space if (child is Space) continue // 确保子 view 已测量 if (child.measuredWidth == 0 || child.measuredHeight == 0) { val widthSpec = if (child.layoutParams.width >= 0) View.MeasureSpec.makeMeasureSpec(child.layoutParams.width, View.MeasureSpec.EXACTLY) else View.MeasureSpec.makeMeasureSpec(group.measuredWidth, View.MeasureSpec.AT_MOST) val heightSpec = if (child.layoutParams.height >= 0) View.MeasureSpec.makeMeasureSpec(child.layoutParams.height, View.MeasureSpec.EXACTLY) else View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED) child.measure(widthSpec, heightSpec) } // 创建骨架色块 val childSkeleton = createAutoSkeleton(child, config).view // 获取子 view 在屏幕中的绝对坐标 val childLocation = IntArray(2) child.getLocationInWindow(childLocation) val childGlobalLeft = childLocation[0] val childGlobalTop = childLocation[1] // 计算相对于 skeletonContainer 的左上角偏移(因为 skeletonContainer 将作为 overlay 显示在同一位置) val leftInParent = childGlobalLeft - parentX val topInParent = childGlobalTop - parentY // 设置 FrameLayout.LayoutParams 并用 margin 定位 val lp = FrameLayout.LayoutParams(child.measuredWidth, child.measuredHeight) lp.leftMargin = leftInParent lp.topMargin = topInParent skeletonContainer.addView(childSkeleton, lp) } return skeletonContainer }给这段代码添加log,我希望看到每次进来这个view group的每个孩子是多大,位置在哪,父亲是谁
09-23
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值