解决自定义控件大小问题

来源:互联网 发布:淘宝网直播怎么入住 编辑:程序博客网 时间:2024/06/09 18:26

对于一些我们不能直接使用的控件,我们通常会重写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_parentretSize = specSize;}else{                              // 如使用wrap_contentretSize = (isWidth? bg_on.getWidth()+padding : bg_on.getHeight()+padding);if(specMode==MeasureSpec.UNSPECIFIED){retSize = Math.min(retSize, specSize);}}        return retSize;}




原创粉丝点击