Android View

1. View绘制机制

1.1 View树的绘制流程

  1. 判断是否需要重新计算视图大小(measure)
  2. 是否重新需要安置视图的位置(layout)
  3. 是否需要重绘(draw)

1.2 measure

mesarue():树形递归过程

作用:为整个view树计算实际大小 -> 设置实际的宽高【mMeasuredHeight, mMeasureWidth】

每个view控件的实际宽高都是由父试图和本身试图决定

**调用链:**ViewRoot根对象的属性mView() -> measure() => View树大小,回调View/ViewGroup对象的onMeasure()方法

实现功能:
1. 设置本View视图最终大小。setMeasuredDimension() –mMeasureHeight/mMeasureWidth
2. 对ViewGroup类型对象的子视图进行遍历的measure()过程,重写onMeasure()

  • 子视图measure()过程 ->调用父ViewGroup.java中的measureChildWithMargins()实现
  • measureChildWithMargins()只是一个过渡层
  • 简单直接的方法:View对象的measure()

LayoutParams继承于Android.View.ViewGroup.LayoutParams相当于一个Layout的信息包,它封装了Layout的位置、高、宽等信息。假设在屏幕上一块区域是由一个Layout占领的,如果将一个View添加到一个Layout中,最好告诉Layout用户期望的布局方式,也就是将一个认可的layoutParams传递进去。通俗地讲LayoutParams类是用于child view(子视图)向parent view(父视图)传达自己的意愿的一个东西(孩子想变成什么样向其父亲说明)。

我们可以通过 LayoutParams来设置view的尺寸,但是仅依靠我们设置的参数,能完全决定view的尺寸吗? 肯定是不能的。因为一个View大小还要受到父ViewGroup的影响。一个view的尺寸大小,既要考虑自己所希望的大小,但是也要考虑父ViewGroup对其所施加的影响。

MeasureSpec其实就是View当中的一个静态工具类,翻译过来就是测量说明书。 代表了View在测试过程中所受到的约束。在View的测量过程中,系统将View自身的 LayoutParams结合父容器所施加的规则转换成对应的MeasureSpec,然后再根据这个MeasureSpec来测量出View的尺寸宽高 。

MeasureSpec代表了一个32位的int类型的值。高两位是SpecMode(测量模式), 低30位是 SpecSize(尺寸规格大小)。将SpecMode和SpecSize通过位操作封装成一个int值,可以减少内存分配,提升效率。 而MeasureSpec提供了打包,解包,toString等方法。可以方便操作。

SpecMode的取值可为以下三种:

  • EXACTLY: 对子View提出了一个确切的建议尺寸(SpecSize);
  • AT_MOST: 子View的大小不得超过SpecSize;
  • UNSPECIFIED: 对子View的尺寸不作限制,通常用于系统内部。

对于父容器(ViewGroup)而言,往往会重载onMeasure函数负责其children的measure工作,重载时不要忘记调用setMeasuredDimension来设置自身的mMeasuredWidth和mMeasuredHeight。
对于子控件(View)而言,通过调用系统默认的onMeasure,即可完成View的测量,当然你也可以重载onMeasure,并调用setMeasuredDimension来设置任意大小的布局。

1.3. Layout

layout():layout阶段的基本思想也是由根View开始,递归地完成整个控件树的布局(layout)工作。

作用:为将整个根据子视图的大小以及布局参数将View树放到合适的位置上

**调用链:**host.layout()开始View树布局,回调给View/ViewGroup类中的layout()
1. layout() -> 设置该View视图位于父视图的坐标轴,mLeft,mTop,mLeft,mBottom(调用setFrame()函数去实现),接下来回调onLayout()方法;
2. view是ViewGroup -> 遍历每个子视图childView,调用该子视图的layout()去设置它的坐标值。

1.4. Draw

作用:标志位DRAWN,每次发起绘图时,为该View添加该标志位,只会重新绘制那些“需要重绘”的视图
ViewRoot – performTraversals() ->draw() 发起绘制该View树,

调用流程:

  1. 绘制背景
  2. 为显示渐变框做一些准备操作
  3. onDraw()【每个View都需要重载该方法,ViewGroup不需要实现该方法】
  4. dispatchDraw () -> 绘制子试图(ViewGroup重写dispatchDraw (),应用程序可以重载父类函数实现具体的功能)
    dispatchDraw()方法内部会遍历每个子视图,调用drawChild()去重新回调每个子视图的draw()方法
  5. 滚动条

invalidate和requestLayout的区别参阅Here

invalidate()

调用draw()过程,假如视图发生大小没有变化就不会调用layout()过程,并且只绘制那些“需要重绘的”视图
谁(View的话,只绘制该View ;ViewGroup,则绘制整个ViewGroup)请求invalidate(),就绘制该视图。

一般引起invalidate()操作的函数如下:

  1. invalidate():请求重新draw(),但只会绘制调用者本身。
  2. setSelection() :请求重新draw(),但只会绘制调用者本身。
  3. setVisibility() : 当View可视状态在INVISIBLE转换VISIBLE时,会间接调用invalidate()方法,继而绘制该View。
  4. setEnabled() : 请求重新draw(),但不会重新绘制任何视图包括该调用者本身。

requestLayout()

只是对View树重新布局layout过程,调用measure()过程 和 layout()过程 ,不会重新绘制任何视图包括该调用者本身。

一般引起requestLayout()操作的函数如下:

  1. setVisibility()方法:
    • 当View的可视状态在INVISIBLE/ VISIBLE 转换为GONE状态时,间接调用requestLayout() 和invalidate()。
    • 同时,由于整个个View树大小发生了变化,会请求measure()过程以及draw()过程,同样地,只绘制需要“重新绘制”的视图。

2. 事件分发机制

Android上面的View是树形结构的,View可能会重叠到一起,当我们点击的地方有多个View都可以响应时,这个点击事件应该给谁呢?为了解决这个问题,就有了事件分发机制。

事件类型具体动作
MotionEvent.ACTION_DOWN按下View(所有事件的开始)
MotionEvent.ACTION_UP抬起View(与DOWN对应)
MotionEvent.ACTION_MOVE滑动View
MotionEvent.ACTION_CANCEL结束事件(非人为原因)

事件链

事件分发的本质是将点击事件(MotionEvent)传递到某个具体的View & 处理的整个过程。

事件分发的顺序:Activity -> ViewGroup -> View,即:1个点击事件发生后,事件先传到Activity、再传到ViewGroup、最终再传到 View。

事件分发过程方法:

源码分析参见Here


参考并感谢

  1. Android View的绘制流程
  2. 深入理解Android之View的绘制流程
  3. androidView树的绘图流程
  4. 从源码看invalidate和requestLayout的区别
  5. Android事件分发机制详解:史上最全面、最易懂
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值