onMeasure

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
	setMeasuredDimension(measureWidth(widthMeasureSpec),
			measureHeight(heightMeasureSpec));
}
// The width of the view, honoring(履行) constraints(约束) from measureSpec
private int measureWidth(int measureSpec) {
	int result = 0;
	int specMode = MeasureSpec.getMode(measureSpec);
	int specSize = MeasureSpec.getSize(measureSpec);
	if (specMode == MeasureSpec.EXACTLY) {
		// We were told how big to be
		result = specSize;
	} else {
		// Measure the text
		result = (int) mTextPaint.measureText(mText) + getPaddingLeft()
				+ getPaddingRight();
		if (specMode == MeasureSpec.AT_MOST) {
			// Respect AT_MOST value if that was what is called for by measureSpec
			result = Math.min(result, specSize); // 不能超过specSize
		}
	}
	return result;
}

思路是这样的:我们首先判断是不是EXACTLY模式,如果是,那就可以直接设置值了,如果不是,我们先按照UNSPECIFIED模式处理,让子布局得到自己想要的最大值,然后判断是否是AT_MOST模式,来做最后的限制。

1、MeasureSpec有3种模式分别是UNSPECIFIED、 EXACTLY、AT_MOST
如果是AT_MOST,specSize 代表的是最大可获得的空间; 
如果是EXACTLY,specSize 代表的是精确的尺寸; 
如果是UNSPECIFIED,对于控件尺寸来说,没有任何参考意义。

2、这些模式和我们平时设置的layout参数fill_parent, wrap_content有什么关系呢?
当设置width或height为fill_parent时,容器在布局时调用子 view的measure方法传入的模式是EXACTLY,因为子view会占据剩余容器的空间,所以它大小是确定的。
当设置为wrap_content时,容器传进去的是AT_MOST, 表示子view的大小最多是多少,这样子view会根据这个上限来设置自己的尺寸。
当子view的大小设置为精确值时,容器传入的是EXACTLY,
而MeasureSpec的UNSPECIFIED模式目前还没有发现在什么情况下使用。 

iew的onMeasure方法默认行为是当模式为UNSPECIFIED时,设置尺寸为mMinWidth(通常为0)或者背景drawable的最小尺寸,当模式为EXACTLY或者AT_MOST时,尺寸设置为传入的MeasureSpec的大小。 

有个观念需要纠正的是,fill_parent应该是子view会占据剩下容器的空间,而不会覆盖前面已布局好的其他view空间,当然后面布局子 view就没有空间给分配了,所以fill_parent属性对布局顺序很重要。以前所想的是把所有容器的空间都占满了,难怪google在2.2版本里把fill_parent的名字改为match_parent.

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
当 Android 中的对话框位置发生更改时,会触发对话框中子视图的 `onMeasure` 方法。如果你想在更改位置后取消子视图的 `onMeasure`,可以尝试以下方法: 1. 在对话框的 `onCreate` 方法中,将对话框的布局参数设置为 `WRAP_CONTENT`。这将使对话框的大小根据其内容自动调整,而不是强制使用固定大小。 2. 在对话框的 `onWindowAttributesChanged` 方法中,将对话框的布局参数设置为屏幕宽度的一半。这将使对话框始终处于屏幕中央,并且不会触发子视图的 `onMeasure` 方法。 以下是示例代码: ```java public class MyDialog extends Dialog { public MyDialog(Context context) { super(context); } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.my_dialog_layout); // 设置对话框的布局参数为 WRAP_CONTENT WindowManager.LayoutParams params = getWindow().getAttributes(); params.width = WindowManager.LayoutParams.WRAP_CONTENT; getWindow().setAttributes(params); } @Override public void onWindowAttributesChanged(WindowManager.LayoutParams params) { super.onWindowAttributesChanged(params); // 设置对话框的布局参数为屏幕宽度的一半 DisplayMetrics metrics = getContext().getResources().getDisplayMetrics(); params.width = metrics.widthPixels / 2; getWindow().setAttributes(params); } } ``` 注意:这是一种简单的解决方案,可能不适用于所有情况。如果子视图仍然需要进行测量,请考虑使用其他方法来优化性能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值