Android~自定义View和事件分发

boolean 表示布尔值

integer 表示整型值

float 表示浮点值

fraction 表示百分数

enum 表示枚举值

flag 表示位运算

–>

  1. attrs.xml文件declare-styleable标签定义及相关属性

  2. 在布局文件中导入自定义的属性集。两种方法

xmlns:custom=“http://schemas.android.com/apk/res/com.example”

xmlns:custom=“http://schemas.android.com/apk/res-auto”

  1. 代码中如何获取自定义属性值

TypedArray arry = context.obtainStyledAttributes(attrs, R.styleable.CustomView);

float size = arry.getDimension(R.styleable.CustomView_size,10f);

// to use

[](()ViewGroup绘制流程

View和ViewGroup绘制流程基本相同,只是ViewGroup除了绘制自己还需要绘制子控件。绘制流程分为测量 ——布局——绘制 主要对应下面三个函数:

  1. onMeasure():测量当前控件大小并为布局提供建议

  2. onLayout():使用layout()函数对所有子控件进行布局

  3. onDraw():根据测量布局的位置绘图

[](()measure流程和MeasureSpec

MeasureSpec是int型数字,但它由两部分组成mode+size,它转换为二进制前两位代表模式后30位代表数值,它有三种模式。MeasureSpec是View的内部类,作用是在Measure的过程中,将View的LayoutParams根据父容器所分发的规则转换成对应的MeasureSpec,最后在onMeasure根据该值确定View的宽高。

  1. UNSPECIFIED: 未指定模式,子View不受父View的限制,子View可以设置任意大小。一般用于系统内部的测量。

  2. EXACTLY:精确模式,对应于match_parent和具体数值,子元素被限定于给定的边界。

  3. AT_MOST:最大模式,对应于wrap_content,父控件给子控件分配的SpecSize。

作为顶层的View,它没有父容器。DecorView的getRootMeasureSpec方法第一个参数windowSize是指窗口尺寸,它的MeasureSpec由自身的LayoutParams和窗口尺寸大小决定。

特别需要注意的是,wrap_content对应AT_MOST,当布局文件中配置为EXACTLY模式时,我们就直接使用该值即可,当模式为AT_MOST,我们还需要将大小设置为我们计算的值,该值应该是包含控件最大值。

View的measure流程 : 先判断有无背景,取mMinWidth和背景的最小宽度的最大值; 再通过measureSpec获取默认大小

protected void onMeas 《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》无偿开源 徽信搜索公众号【编程进阶路】 ure(int widthMeasureSpec, int heightMeasureSpec) {

setMeasuredDimension(getDefaultSize(getSuggestedMinimumWidth(), widthMeasureSpec),

getDefaultSize(getSuggestedMinimumHeight(), heightMeasureSpec));

}

public static int getDefaultSize(int size, int measureSpec) {

int result = size;

int specMode = MeasureSpec.getMode(measureSpec);

int specSize = MeasureSpec.getSize(measureSpec);

switch (specMode) {

case MeasureSpec.UNSPECIFIED:

result = size;

break;

case MeasureSpec.AT_MOST:

case MeasureSpec.EXACTLY:

result = specSize;

break;

}

return result;

}

protected int getSuggestedMinimumWidth() {

return (mBackground == null) ? mMinWidth : max(mMinWidth, mBackground.getMinimumWidth());

}

ViewGroup源码中无onMeasure方法,它的measure流程 :直接遍历测量子View的MeasureSpec,measureChild中则是先获取自己的LayoutParams ,再计算自己的getChildMeasureSpec。

protected void measureChildren(int widthMeasureSpec, int heightMeasureSpec) {

final int size = mChildrenCount;

final View[] children = mChildren;

for (int i = 0; i < size; ++i) {

final View child = children[i];

if ((child.mViewFlags & VISIBILITY_MASK) != GONE) {

measureChild(child, widthMeasureSpec, heightMeasureSpec);

}

}

}

protected void measureChild(View child, int parentWidthMeasureSpec,

int parentHeightMeasureSpec) {

final LayoutParams lp = child.getLayoutParams();

final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,

mPaddingLeft + mPaddingRight, lp.width);

final int childHeightMeasureSpec = getChildMeasureSpec(parentHeightMeasureSpec,

mPaddingTop + mPaddingBottom, lp.height);

child.measure(childWidthMeasureSpec, childHeightMeasureSpec);

}

[](()layout流程

View中的layout是用来确定自身的位置。调用层级是layout调用setFrame确定该View在父容器中的位置,最后才调用onLayout。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android中的点击事件分发是指在触摸屏幕上的某个位置按下并释放手指后,系统是如何将该事件分发给目标View的。点击事件分发涉及两个主要阶段:事件捕获和事件冒泡。 在事件捕获阶段,Android系统从顶层的父级View开始向下遍历整个View树,判断每个View是否需要处理该事件事件捕获是从父级View到子级View的遍历过程,直到找到最内层的可点击View或者没有View处理该事件为止。 接下来是事件冒泡阶段,触摸事件从最内层的View开始向上返回,依次经过包含它的每个父级View,并判断每个View是否需要处理该事件事件冒泡是从最内层View到最外层View的遍历过程,直到找到一个View处理该事件或者达到最外层的父级View为止。 在分发点击事件的过程中,每个View都可以通过重写相关的事件处理方法来拦截并处理事件,比如onTouchEvent()、onInterceptTouchEvent()和dispatchTouchEvent()等。这些方法可以根据需要进行定制,从而实现对事件的控制和处理。 事件分发机制是由ViewGroup和View共同实现的,通过相互协作完成点击事件的传递和处理。在此过程中,还有一些特殊情况需要考虑,比如View的可点击区域、事件的传递方向和拦截以及多个View同时处理点击事件等等。 总结来说,Android中的点击事件分发是通过事件捕获和事件冒泡的方式实现的,通过ViewGroup和View相互协作完成点击事件的传递和处理。开发者可以通过重写相关方法来控制和处理事件,实现自定义的交互逻辑。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值