android 自定义 view 和 ViewGroup

时间:2022-08-30 08:04:08

ViewGroup的职能为:给childView计算出建议的宽和高和测量模式 ;决定childView的位置;为什么只是建议的宽和高,而不是直接确定呢,别忘了childView宽和高可以设置为wrap_content,这样只有childView才能计算出自己的宽和高。

View的职责:根据测量模式和ViewGroup给出的建议的宽和高,计算出自己的宽和高;同时还有个更重要的职责是:在ViewGroup为其指定的区域内绘制自己的形态。

在自定义view中:

1、自定义View的属性

  通过attrs.xml 添加

1 <declare-styleable name="CustomTextView">  
2    <attr name="titleText" />  
3    <attr name="titleTextColor" />  
4    <attr name="titleTextSize" />  
5 </declare-styleable>  

2、在View的构造方法中获得我们自定义的属性

  通过Context.obtainStyledAttributes获取xml属性。

3、重写onMesure 

  android提供MeasureSpec类帮助测量view。MeasureSpec是一个32位的int值,其中高两位为测量模式,低30位为测量大小。

  1、EXACTLY

    精确值模式,当我们的控件的layout_width属性和layout_height属性指定为具体数值时,如android:layout_width="100dp"或者为match_parent时系统使用的是EXACTLY模式。

    父控件可以通过MeasureSpec.getSize(measureSpec)直接得到子控件的尺寸。

    父View给自定义View确定了一个范围,在这个范围内,自定义view的大小是给出的具体的值,比如 width =100dp,height=200dp,但是如果给出的任何一个数值超过了父View的限制值,他最大是父View的限制值

  2、AT_MOST

    最大值模式,当控件layout_width属性和layout_height属性为wrap_content时,控件大小随控件子控件或内容变化,此时控件的尺寸只要不超过父控件允许的最大尺寸即可。

    这种模式下,父控件无法确定子 View 的尺寸,只能由子控件自己根据需求去计算自己的尺寸,这种模式就是我们自定义视图需要实现测量逻辑的情况。

  3、UNSPECIFIED

    不指定大小测量模式。父视图不对子视图有任何约束,它可以达到所期望的任意尺寸,比如 ListView、ScrollView,一般自定义 View 中用不到。

    View类默认的onMeasure()方法只支持EXACTLY模式,如果想让控件支持wrap_content属性就必须重写onMeasure()方法

    重写onMeasure()方法后需调用setMeasuredDimension(int measuredWidth,int measuredHeight)

  设置了WRAP_CONTENT时,我们需要自己进行测量,必须重写onMeasure()方法

  

// ViewRootImpl#getRootMeasureSpec源码:
private static int getRootMeasureSpec(int windowSize, int rootDimension) {
        int measureSpec;
        switch (rootDimension) {

        case ViewGroup.LayoutParams.MATCH_PARENT:
            // Window can't resize. Force root view to be windowSize.
            measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.EXACTLY);
            break;
        case ViewGroup.LayoutParams.WRAP_CONTENT:
            // Window can resize. Set max size for root view.
            measureSpec = MeasureSpec.makeMeasureSpec(windowSize, MeasureSpec.AT_MOST);
            break;
        default:
            // Window wants to be an exact size. Force root view to be that size.
            measureSpec = MeasureSpec.makeMeasureSpec(rootDimension, MeasureSpec.EXACTLY);
            break;
        }
        return measureSpec;
    }

 

4、重写onDraw

*********主要View的执行过程*********:

(1)构造方法 (2)onFinishInflate (3)onSizeChanged(4)onDraw

在自定义ViewGroup中:

1. onMeasure中计算childView的测量值以及模式,以及设置自己的宽和高

2. onLayout对其所有childView进行定位(设置childView的绘制区域)

  通过getChildCount()获取总子view,getChildAt获取childview调用各自的layout(int, int, int, int)方法。

ViewGroup不会执行onDraw说明:

1)ViewGroup默认情况下,会被设置成WILL_NOT_DRAW,这是从性能考虑,这样一来,onDraw就不会被调用了。

2)如果我们要重要一个ViweGroup的onDraw方法,有两种方法:

        1,在构造函数里面,给其设置一个颜色,如#00000000。

        2,在构造函数里面,调用setWillNotDraw(false),去掉其WILL_NOT_DRAW flag。

 

注意,自定义的View在使用的时候一定要写出完整的包名,不然系统将无法找到这个View。

xmlns:custom="http://schemas.android.com/apk/res/com.example.customview01" 

 

Android中实现view的更新有两组方法

一组是invalidate,另一组是postInvalidate,其中前者是在UI线程自身中使用,而后者在非UI线程中使用。 

前面要利用Handler结合使用和利用postInvalidate()来实现在线程中刷新界面。 

1,利用invalidate()刷新界面 
  实例化一个Handler对象,并重写handleMessage方法调用invalidate()实现界面刷新;而在线程中通过sendMessage发送界面更新消息。 

2,使用postInvalidate()刷新界面 
  使用postInvalidate则比较简单,不需要handler,直接在线程中调用postInvalidate即可。 (*****源码也是通过handler去执行*****)