美文网首页
自定义View

自定义View

作者: shuixingge | 来源:发表于2016-05-11 15:05 被阅读54次

    4.1 初始ViewRoot和DecorView
    View的绘制流程是从ViewRoot的performTraversal的开始的。经过measure,layout,draw三个过程。
    performTraversal会依次调用performMeasure,performLayout,performDraw来完成顶级View的measure,layout和draw过程。performMeasure方法中会调用measure方法,在measure方法中又会调用onMeasure方法,在onMeasure方法中会对所有的子元素进行measure过程,这个时候measure流程就从父容器传递到子元素了,这样就完成了一次measure过程,layout和draw的过程类似。
    (1)measure过程决定了View的宽高,几乎在所有情况下,这个宽高都等于View最终的宽高。getMeasuredHeight(),getMeasuredWidth()来得到测量宽高。
    (2)layout过程决定了View四个顶点的位置,和View实际的宽高。通过getHeight()和getWidth()来得到实际的宽高。
    (3)draw过程决定view的显示。
    DecorView是一个FrameLayout,包含了一个竖直方向的LinearLayout,上面是标题栏,下面是内容。
    4.1 理解MeasureSpec
    (1)MeasureSpec和LayoutParams的对应关系。
    在View测量的时候,系统会将LayoutParams在父容器的约束下转换成MeasureSpec,在根据MeasureSpec来决定View测量后的宽高。
    MeasureSpec不是唯一由LayoutParams决定的,而是和父容器一起决定的。对于DecorView,它的MeasureSpec是由窗口的大小和自身的LayoutParams确定的。
    (2)普通view的MeasureSpec的创建规则
    当View采用固定宽高时,不管父容器采用什么MeasureSpec,View的MeasureSpec都是精确模式,大小等于LayoutParams中的大小。
    当View采用match_parent时,如果父容器是精确模式,那么view也是精确模式,大小是父容器剩余的大小。如果父容器是最大模式,那么view也是最大模式,大小不超过父容器的剩余空间。
    当View采用wrap_content时,不管父容器是什么模式,View都是最大模式,大小是不超过父容器剩余的大小。
    4.3 View的工作流程
    (1)measure过程
    view的测量过程由measure方法(final)完成,在measure方法中又会调用onMeasure方法,onMeasure会调用setMeasuredDimension方法,setMeasuredDimension会设置View宽高的测量值。当View的SpecMode是AT_MOST和EXACTLY时,getDefaultSize返回的是measureSpec中的specSize的大小,而当View的SpecMode是UNSPECIFIED的时候
    返回的是size,是getSuggestedMinimumWidth的返回值,如果View没有设置背景,那么返回的是android:midWidth这个属性,如果设置了背景,返回的是android:minWidth和背景最小宽度中两者的的最大值。

    Paste_Image.png

    (2)ViewGroup的measure过程
    viewGroup除了要完成自己的measure以外,还会遍历去调用所有子元素的measure方法,各个子元素在递归完成此过程。ViewGroup是一个抽象类,因此它没有重写onMeasure方法。在onMeasure方法中拿到的测量宽高是不准确的,在onLayout中获得测量宽高或者最终的宽高。view的measure过程和Activity的生命周期方法不是同步执行的,因此无法保证Activity执行了onCreate、onStart、onResume时某个view已经测量完毕了。如果view还没有测量完毕,那么获得的宽高就都是0。下面是四种解决该问题的方法:
    一:onWindowFocusChanged():View已经初始化完毕,宽高已经准备好了。
    二:view.post(runnable):通过post可以将一个runnable投射到消息队列的尾部,然后等待Looper调用此runnable的时候,View就已经初始化好了。
    三:ViewTreeObserver:使用ViewTreeObserver的众多回调方法可以完成这个功能,比如使用onGlobalLayoutListener接口,当view树的状态发生改变或者view树内部的view的可见性发生改变时,onGlobalLayout方法将被回调。伴随着view树的状态改变,这个方法也会被多次调用。
    在view的默认实现中,view的测量宽高和最终宽高是相等的,只不过测量宽高形成于measure过程,而最终宽高形成于layout过程。
    (3)layout过程
    ViewGroup的位置被确定以后,它在onLayout中会遍历所有的子元素并调用其layout方法,在layout方法中OnLayout方法又会被调用。layout方法的流程:首先会通过setFrame方法来设定View的四个顶点,View一旦确定,View在父容器中的位置也被确定了,接着会调用onLayout方法,这个方法的用途是确定子元素的位置。onLayout的具体实现和具体的布局有关。
    (4)draw过程
    1.绘制背景:background.draw(canvas);
    2.绘制自己:onDraw();
    3.绘制children:dispatchDraw;
    4.绘制装饰:onDrawScrollBars。
    4.4 自定义view
    (1)继承view重写onDraw方法需要自己支持wrap_content,并且padding也要自己处理。继承特定的View例如TextView不需要考虑。
    继承自ViewGroup要在onMeasure和onLayout中考虑padding和子元素的margin对其造成的影响。
    (2)尽量不要在View中使用Handler,因为view内部本身已经提供了post系列的方法,完全可以替代Handler的作用。
    (3)view中如果有线程或者动画,需要在onDetachedFromWindow方法中及时停止。
    (4)处理好view的滑动冲突情况。
    如何处理wrap_content,原因?
    如果在View在布局中使用wrap_content,那么它的specMode是AT_MOST,在这种模式下,它的宽高等于specSize,而specSize等于parentSize。parentSize是父容器目前剩余空间的大小。和match_parent一致。

    相关文章

      网友评论

          本文标题:自定义View

          本文链接:https://www.haomeiwen.com/subject/zglurttx.html