1、自定义控件首先定义一个类继承View
有时,Android系统控件无法满足我们的需求,因此有必要自定义View。具体方法参见官方开发文档: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方法传入的模式。
我们来看下程序的代码:
xml布局文件:
<?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity_main" android:layout_width="match_parent" android:layout_height="match_parent" android:paddingBottom="@dimen/activity_vertical_margin" android:paddingLeft="@dimen/activity_horizontal_margin" android:paddingRight="@dimen/activity_horizontal_margin" android:paddingTop="@dimen/activity_vertical_margin" tools:context="im.weiyuan.com.zidingyikongjian.MainActivity"> <im.weiyuan.com.zidingyikongjian.MesureView android:layout_width="100px" android:layout_height="200px" /> </RelativeLayout>
我们来看看类的代码,该代码继承自View类
package im.weiyuan.com.zidingyikongjian; import android.content.Context; import android.util.AttributeSet; import android.util.Log; import android.view.View; /** * Created by wei.yuan on 2017/6/2. */ public class MesureView extends View { /** * * new MesureView(context)的时候调用该构造函数 * */ public MesureView(Context context) { super(context); } /** * 在布局中引用 * * */ public MesureView(Context context, AttributeSet attrs) { super(context, attrs); } /** * 在布局中引用 * * */ public MesureView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); } /** * 在布局中引用 * * */ public MesureView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { super(context, attrs, defStyleAttr, defStyleRes); } /** * * * */ @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(widthMeasureSpec, heightMeasureSpec); // 第一步提取出具体对象的测量模式和大小 int size = MeasureSpec.getSize(widthMeasureSpec); int mode = MeasureSpec.getMode(widthMeasureSpec); Log.d("123456",size+""); if(mode == MeasureSpec.EXACTLY){ //默认就是精确的模式 Log.d("123456",mode+""); } } }
我们来看看日志打印的效果:
MeasureSpec.getSize获得对应的控件的大小:单位是px,这里上面测量的是控件的宽度,在xml布局里面设置的是100px,这里打印就是100px
这里控件在xml指定的是具体的100px大小值,这里获得的mode就是精确的模式、
setMeasuredDimension这个方法,这个方法决定了当前View的大小
现在我们在xml中指定了控件的宽度是100px,高度是100px
现在我们要通过代码指定控件的宽度和高度可以使用函数
setMeasuredDimension
现在我们的首先的像素是1080*1920,我们想要这个控件填充我们的整个屏幕可以设置下面的代码:
package im.weiyuan.com.zidingyikongjian; import android.content.Context; import android.util.AttributeSet; import android.util.Log; import android.view.View; /** * Created by wei.yuan on 2017/6/2. */ public class MesureView extends View { /** * * new MesureView(context)的时候调用该构造函数 * */ public MesureView(Context context) { super(context); } /** * 在布局中引用 * * */ public MesureView(Context context, AttributeSet attrs) { super(context, attrs); } /** * 在布局中引用 * * */ public MesureView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); } /** * 在布局中引用 * * */ public MesureView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) { super(context, attrs, defStyleAttr, defStyleRes); } /** * * * */ @Override protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) { super.onMeasure(widthMeasureSpec, heightMeasureSpec); // 第一步提取出具体对象的测量模式和大小 int size = MeasureSpec.getSize(widthMeasureSpec); int mode = MeasureSpec.getMode(widthMeasureSpec); Log.d("123456",size+""); if(mode == MeasureSpec.EXACTLY){ //默认就是精确的模式 Log.d("123456",mode+""); } //设置控制的大小 setMeasuredDimension(1080,1920);//单位是像素 } }