解决自定义控件大小问题

对于一些我们不能直接使用的控件,我们通常会重写View来自定义功能和样式,这时放在布局文件里,设置为wrap_content后也是fill_parent的样式,不能在一个页面内同时显示两个自定义的组件,这时我们就需要重写onMeasure()方法。具体重写自定义组件的方法参见官方开发文档:http://developer.android.com/guide/topics/ui/custom-components.html


一般来说,自定义控件都会去重写View的onMeasure方法,因为该方法指定该控件在屏幕上的大小。
protected void onMeasure (int widthMeasureSpec, int heightMeasureSpec)


onMeasure传入的两个参数是由上一层控件传入的大小,有多种情况,重写该方法时需要对计算控件的实际大小,然后调用setMeasuredDimension(int, int)设置实际大小。
onMeasure传入的widthMeasureSpec和heightMeasureSpec不是一般的尺寸数值,而是将模式和尺寸组合在一起的数值。我们需要通过int mode = MeasureSpec.getMode(widthMeasureSpec)得到模式,用int size = MeasureSpec.getSize(widthMeasureSpec)得到尺寸。


mode共有三种情况,取值分别为MeasureSpec.UNSPECIFIED, MeasureSpec.EXACTLY, MeasureSpec.AT_MOST。
MeasureSpec.EXACTLY是精确尺寸,当我们将控件的layout_width或layout_height指定为具体数值时如andorid:layout_width="50dip",或者为FILL_PARENT是,都是控件大小已经确定的情况,都是精确尺寸。
MeasureSpec.AT_MOST是最大尺寸,当控件的layout_width或layout_height指定为WRAP_CONTENT时,控件大小一般随着控件的子空间或内容进行变化,此时控件尺寸只要不超过父控件允许的最大尺寸即可。因此,此时的mode是AT_MOST,size给出了父控件允许的最大尺寸。
MeasureSpec.UNSPECIFIED是未指定尺寸,这种情况不多,一般都是父控件是AdapterView,通过measure方法传入的模式。


因此,在重写onMeasure方法时要根据模式不同进行尺寸计算。下面代码就是一种比较典型的方式:

// 例如重写一个可滑动的SlipButton,bg_on为“开”时的背景
Bitmap bg_on;
...
// 初始化等其他操作。
...

@Override 
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { 
	// 这里要计算一下控件的实际大小,然后调用setMeasuredDimension来设置
  int width = this.getMeasuredSize(widthMeasureSpec, true);
  int height = this.getMeasuredSize(heightMeasureSpec, false);
  setMeasuredDimension(width, height);
} 

/**
	* 计算控件的实际大小
	* @param length onMeasure方法的参数,widthMeasureSpec或者heightMeasureSpec
	* @param isWidth 是宽度还是高度
	* @return int 计算后的实际大小
	*/
private int getMeasuredSize(int length, boolean isWidth){
	// 模式
	int specMode = MeasureSpec.getMode(length);
	// 尺寸
	int specSize = MeasureSpec.getSize(length);
	// 计算所得的实际尺寸,要被返回
	int retSize = 0;        
	// 得到两侧的padding(留边)
	int padding = (isWidth? getPaddingLeft()+getPaddingRight():getPaddingTop()+getPaddingBottom());
        
	// 对不同的指定模式进行判断
	if(specMode==MeasureSpec.EXACTLY){  // 显式指定大小,如40dp或fill_parent
		retSize = specSize;
	}else{                              // 如使用wrap_content
		retSize = (isWidth? bg_on.getWidth()+padding : bg_on.getHeight()+padding);
		if(specMode==MeasureSpec.UNSPECIFIED){
		retSize = Math.min(retSize, specSize);
		}
	}        

	return retSize;
}




你可能感兴趣的:(自定义,android)