Android自定义控件(十一)——自定义ViewGroup实现LinearLayout

简介: Android自定义控件(十一)——自定义ViewGroup实现LinearLayout

ViewGroup的绘制流程


要自定以ViewGroup,我们首先需要了解ViewGroup的绘制流程,其实View与ViewGroup绘制基本相同,只是在ViewGroup中,不仅仅要绘制自己,还要绘制其中的子控件,所以ViewGroup的绘制流程分为三步:测量,布局,绘制,分别对应onMeasure(),onLayout(),onDraw()。


1.onMeasure():测量当前控件的大小,为正式布局提供建议,注意仅仅只是建议,至于用不用看onLayout()。


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


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


这里onDraw()就不再赘述了,前面自定义的所有View()基本都讲解过如何使用onDraw(),本博文重点介绍onMeasure()和onLayout()函数。


onMeasure()函数与MeasureSpec

首先,我们来看看onMeasure() 函数的定义:


protected void onMeasure(int widthMeasureSpec,int heightMeasureSpec)

这里主要传进去两个参数,分别是widthMeasureSpec和heightMeasureSpec。它们是父类传递过来给当前ViewGroup的一个建议值,即想把当前ViewGroup的尺寸设置为宽widthMeasureSpec,高heightMeasureSpec。


虽然他们两个是int类型,但其实他们是由mode+size两部分组成的,转换位二进制都是32位的,前2位代表模式mode,后30位代表数值。


模式分类

既然说到mode模式,我们来看看它的三种分类:


(1)UNSPECIFIED(未指定):父元素不对子元素施加任何束缚,子元素可以得到任何想要的数值。


(2)EXACTLY(完全):父元素决定子元素的确切大小,子元素将被限定在给定的边界里而忽略它本身的大小。


(3)AT_MOST(至多):子元素至多达到指定大小的值。


既然提到了模式,很显然,我们提取模式进行判断,就需要听过与运算得到,这里Android给我们提供了一个简单的方法,直接提取:

MeasureSpec.getMode(int spec)//获取模式
MeasureSpec.getSize(int spec)//获取数值


那么代码中使用起来的代码就是这样:

int measureWidth=MeasureSpec.getSize(widthMeasureSpec)
int measureWidhtMode=MeasureSpec.getMode(widthMeasureSpec)
//heightMeasureSpec同样如此使用。


如何使用模式

我们先来看看一般我们在XML中如何定义控件的宽高的,有如下三种方式:


(1)warp_content:对应模式MeasureSpec.AT_MOST。


(2)match_parent:对应模式的MeasureSpec.EXACTLY。


(2)具体值(比如设置60px,60dp等):对应模式的MeasureSpec.EXACTLY。


我们从这里看出来,一般我们用不到MeasureSpec.UNSPECIFIED模式。但是我们这里需要注意的是,如果我们设置为MeasureSpec.EXACTLY模式,就不必设置我们计算的大小数值,因为用户已经指定,而设置为MeasureSpec.AT_MOST(warp_content)就需要我们设置具体数值。所以,我们自定义ViewGroup的onMeasure()函数一般都是这样的:

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    int measureWidth=MeasureSpec.getSize(widthMeasureSpec);
    int measureHeight=MeasureSpec.getSize(heightMeasureSpec);
    int measureWidhtMode=MeasureSpec.getMode(widthMeasureSpec);
    int measureHeightMode=MeasureSpec.getMode(heightMeasureSpec);
    //这里计算width,height
    setMeasuredDimension((measureWidhtMode==MeasureSpec.EXACTLY)?measureWidth:width,(measureHeightMode==MeasureSpec.EXACTLY)?measureHeight:height);
    }


如果等于MeasureSpec.EXACTLY就不需要进行计算设置,如果是MeasureSpec.AT_MOST(warp_content)就需要计算控件大小的步骤。


onLayout()函数

前面已经说过了,onLayout()函数是实现所有子控件布局的函数,需要注意的是,这里是实现所有子控件的布局,至于自己我们后面会介绍。我们先来看看onLayout()函数的定义:

protected abstract void onLayout(boolean changed, int l, int t, int r, int b); 

可以看到这是一个抽象函数,说明只要你需要自定义ViewGroup,就必须实现该函数,想我们后面自定义ViewGroup实现LinearLayout一样,都需要重写这个函数,然后按照自己的规则,对子控件进行布局。


自定义ViewGroup实现LinearLayout


我们假设,我们需要自定义的ViewGroup是一个垂直布局,所以我们知道,整体控件的高就是子控件高的和,宽度就是子控件中最宽的哪个,所以我们的onMeasure()函数实现如下:

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    int measureWidth=MeasureSpec.getSize(widthMeasureSpec);
    int measureHeight=MeasureSpec.getSize(heightMeasureSpec);
    int measureWidhtMode=MeasureSpec.getMode(widthMeasureSpec);
    int measureHeightMode=MeasureSpec.getMode(heightMeasureSpec);
    //这里计算width,height
    int height=0;
    int width=0;
    int count=getChildCount();
    for(int i=0;i<count;i++){
        //测量子控件
        View child=getChildAt(i);
        measureChild(child,widthMeasureSpec,heightMeasureSpec);
        int childWidth=child.getMeasuredWidth();
        int childHeight=child.getMeasuredHeight();
        height+=childHeight;//高度叠加
        width=Math.max(width,childWidth);//宽度取最大
    }
    setMeasuredDimension((measureWidhtMode==MeasureSpec.EXACTLY)?measureWidth:width,(measureHeightMode==MeasureSpec.EXACTLY)?measureHeight:height);
    }


可以看到,我们实现的原理,基本与上面讲解的一致,获取子控件最宽的宽度设置为整体ViewGroup的宽度,设置ViewGroup的高度为子控件高度和。因为我们的高度宽度在XML中都设置为了warp_content,这里就需要我们自己计算。(XML代码最后)


接着,就是实现我们的onLayout()函数,因为我们说了我们是实现垂直布局的LinearLayout,所以我们需要在这个函数中布局子控件的位置。代码如下:

protected void onLayout(boolean changed, int l, int t, int r, int b) {
    int top=0;
    int count=getChildCount();
    for(int i=0;i<count;i++){
        View child=getChildAt(i);
        int childWidth=child.getMeasuredWidth();
        int childHeight=child.getMeasuredHeight();
        child.layout(0,top,childWidth,childHeight);
        top+=childHeight;
    }
}


因为我们是垂直布局,也没有设置什么pading,margin,所以左上角坐标就只有top在变化叠加,也就是加上第一个子控件的高度就是第二个子控件的top。我们的XML代码如下:

<?xml version="1.0" encoding="utf-8"?>
<com.liyuanjinglyj.customviewgroup.MyViewGroup xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:background="@android:color/holo_red_dark"
    tools:context=".MainActivity">
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="我是第一个子控件"/>
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="我是第二个子控件"/>
    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="我是第三个子控件(但我加长了)"/>
</com.liyuanjinglyj.customviewgroup.MyViewGroup>


但是我们还是需要注意一下getMeasuredWidth()与getWidth()区别,getMeasuredWidth()函数在onMeasure()过程结束后,就可以获取到宽度值,而getWidth()函数要在onLayout()过程结束后才能获取到宽度值,所以我们上面都使用getMeasuredWidth()。


而且getMeasuredWidth()函数是通过setMeasuredDimension()函数进行设置的,getWidth()函数则是通过layout()函数来设置的。所以我们在前面自定义的所有View中都是在onDraw()中使用getWidth(),因为其他地方必须等onMeasure()与onLayout()指定完后才能获取到。


本文Github下载地址:点击下载

相关文章
|
4月前
|
Android开发 UED 计算机视觉
Android自定义view之线条等待动画(灵感来源:金铲铲之战)
本文介绍了一款受游戏“金铲铲之战”启发的Android自定义View——线条等待动画的实现过程。通过将布局分为10份,利用`onSizeChanged`测量最小长度,并借助画笔绘制动态线条,实现渐变伸缩效果。动画逻辑通过四个变量控制线条的增长与回退,最终形成流畅的等待动画。代码中详细展示了画笔初始化、线条绘制及动画更新的核心步骤,并提供完整源码供参考。此动画适用于加载场景,提升用户体验。
401 5
Android自定义view之线条等待动画(灵感来源:金铲铲之战)
|
4月前
|
Android开发
Android自定义view之利用PathEffect实现动态效果
本文介绍如何在Android自定义View中利用`PathEffect`实现动态效果。通过改变偏移量,结合`PathEffect`的子类(如`CornerPathEffect`、`DashPathEffect`、`PathDashPathEffect`等)实现路径绘制的动态变化。文章详细解析了各子类的功能与参数,并通过案例代码展示了如何使用`ComposePathEffect`组合效果,以及通过修改偏移量实现动画。最终效果为一个菱形图案沿路径运动,源码附于文末供参考。
|
4月前
|
Android开发 开发者
Android自定义view之利用drawArc方法实现动态效果
本文介绍了如何通过Android自定义View实现动态效果,重点使用`drawArc`方法完成圆弧动画。首先通过`onSizeChanged`进行测量,初始化画笔属性,设置圆弧相关参数。核心思路是不断改变圆弧扫过角度`sweepAngle`,并调用`invalidate()`刷新View以实现动态旋转效果。最后附上完整代码与效果图,帮助开发者快速理解并实践这一动画实现方式。
129 0
|
4月前
|
Android开发 数据安全/隐私保护 开发者
Android自定义view之模仿登录界面文本输入框(华为云APP)
本文介绍了一款自定义输入框的实现,包含静态效果、hint值浮动动画及功能扩展。通过组合多个控件完成界面布局,使用TranslateAnimation与AlphaAnimation实现hint文字上下浮动效果,支持密码加密解密显示、去除键盘回车空格输入、光标定位等功能。代码基于Android平台,提供完整源码与attrs配置,方便复用与定制。希望对开发者有所帮助。
|
4月前
|
XML Java Android开发
Android自定义view之网易云推荐歌单界面
本文详细介绍了如何通过自定义View实现网易云音乐推荐歌单界面的效果。首先,作者自定义了一个圆角图片控件`MellowImageView`,用于绘制圆角矩形图片。接着,通过将布局放入`HorizontalScrollView`中,实现了左右滑动功能,并使用`ViewFlipper`添加图片切换动画效果。文章提供了完整的代码示例,包括XML布局、动画文件和Java代码,最终展示了实现效果。此教程适合想了解自定义View和动画效果的开发者。
207 65
Android自定义view之网易云推荐歌单界面
|
4月前
|
XML 前端开发 Android开发
一篇文章带你走近Android自定义view
这是一篇关于Android自定义View的全面教程,涵盖从基础到进阶的知识点。文章首先讲解了自定义View的必要性及简单实现(如通过三个构造函数解决焦点问题),接着深入探讨Canvas绘图、自定义属性设置、动画实现等内容。还提供了具体案例,如跑马灯、折线图、太极图等。此外,文章详细解析了View绘制流程(measure、layout、draw)和事件分发机制。最后延伸至SurfaceView、GLSurfaceView、SVG动画等高级主题,并附带GitHub案例供实践。适合希望深入理解Android自定义View的开发者学习参考。
504 84
|
4月前
|
前端开发 Android开发 UED
讲讲Android为自定义view提供的SurfaceView
本文详细介绍了Android中自定义View时使用SurfaceView的必要性和实现方式。首先分析了在复杂绘制逻辑和高频界面更新场景下,传统View可能引发卡顿的问题,进而引出SurfaceView作为解决方案。文章通过Android官方Demo展示了SurfaceView的基本用法,包括实现`SurfaceHolder.Callback2`接口、与Activity生命周期绑定、子线程中使用`lockCanvas()`和`unlockCanvasAndPost()`方法完成绘图操作。
102 3
|
4月前
|
Android开发 开发者
Android自定义view之围棋动画(化繁为简)
本文介绍了Android自定义View的动画实现,通过两个案例拓展动态效果。第一个案例基于`drawArc`方法实现单次动画,借助布尔值控制动画流程。第二个案例以围棋动画为例,从简单的小球直线运动到双向变速运动,最终实现循环动画效果。代码结构清晰,逻辑简明,展示了如何化繁为简实现复杂动画,帮助读者拓展动态效果设计思路。文末提供完整源码,适合初学者和进阶开发者学习参考。
Android自定义view之围棋动画(化繁为简)
|
4月前
|
Java Android开发 开发者
Android自定义view之围棋动画
本文详细介绍了在Android中自定义View实现围棋动画的过程。从测量宽高、绘制棋盘背景,到创建固定棋子及动态棋子,最后通过属性动画实现棋子的移动效果。文章还讲解了如何通过自定义属性调整棋子和棋盘的颜色及动画时长,并优化视觉效果,如添加渐变色让白子更明显。最终效果既可作为围棋动画展示,也可用作加载等待动画。代码完整,适合进阶开发者学习参考。
|
4月前
|
传感器 Android开发 开发者
Android自定义view之3D正方体
本文介绍了如何通过手势滑动操作实现3D正方体的旋转效果,基于Android自定义View中的GLSurfaceView。相较于使用传感器控制,本文改用事件分发机制(onTouchEvent)处理用户手势输入,调整3D正方体的角度。代码中详细展示了TouchSurfaceView的实现,包括触控逻辑、OpenGL ES绘制3D正方体的核心过程,以及生命周期管理。适合对Android 3D图形开发感兴趣的开发者学习参考。