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下载地址:点击下载

相关文章
|
2月前
|
搜索推荐 Android开发 开发者
安卓应用开发中的自定义控件实践
在安卓应用开发的广阔天地中,自定义控件如同璀璨的星辰,点亮了用户界面设计的夜空。它们不仅丰富了交互体验,更赋予了应用独特的个性。本文将带你领略自定义控件的魅力,从基础概念到实际应用,一步步揭示其背后的原理与技术细节。我们将通过一个简单的例子——打造一个具有独特动画效果的按钮,来展现自定义控件的强大功能和灵活性。无论你是初学者还是资深开发者,这篇文章都将为你打开一扇通往更高阶UI设计的大门。
|
3月前
|
缓存 前端开发 Android开发
安卓开发中的自定义视图:从零到英雄
【10月更文挑战第42天】 在安卓的世界里,自定义视图是一块画布,让开发者能够绘制出独一无二的界面体验。本文将带你走进自定义视图的大门,通过深入浅出的方式,让你从零基础到能够独立设计并实现复杂的自定义组件。我们将探索自定义视图的核心概念、实现步骤,以及如何优化你的视图以提高性能和兼容性。准备好了吗?让我们开始这段创造性的旅程吧!
56 1
|
3月前
|
前端开发 Android开发 UED
安卓应用开发中的自定义控件实践
【10月更文挑战第35天】在移动应用开发中,自定义控件是提升用户体验、增强界面表现力的重要手段。本文将通过一个安卓自定义控件的创建过程,展示如何从零开始构建一个具有交互功能的自定义视图。我们将探索关键概念和步骤,包括继承View类、处理测量与布局、绘制以及事件处理。最终,我们将实现一个简单的圆形进度条,并分析其性能优化。
|
3月前
|
搜索推荐 前端开发 Android开发
安卓应用开发中的自定义视图实现
【10月更文挑战第30天】在安卓开发的海洋中,自定义视图是那抹不可或缺的亮色,它为应用界面的个性化和交互体验的提升提供了无限可能。本文将深入探讨如何在安卓平台创建自定义视图,并展示如何通过代码实现这一过程。我们将从基础出发,逐步引导你理解自定义视图的核心概念,然后通过一个实际的代码示例,详细讲解如何将理论应用于实践,最终实现一个美观且具有良好用户体验的自定义控件。无论你是想提高自己的开发技能,还是仅仅出于对安卓开发的兴趣,这篇文章都将为你提供价值。
|
3月前
|
Android开发 开发者 UED
安卓开发中自定义View的实现与性能优化
【10月更文挑战第28天】在安卓开发领域,自定义View是提升应用界面独特性和用户体验的重要手段。本文将深入探讨如何高效地创建和管理自定义View,以及如何通过代码和性能调优来确保流畅的交互体验。我们将一起学习自定义View的生命周期、绘图基础和事件处理,进而探索内存和布局优化技巧,最终实现既美观又高效的安卓界面。
69 5
|
4月前
|
缓存 搜索推荐 Android开发
安卓开发中的自定义控件基础与进阶
【10月更文挑战第5天】在Android应用开发中,自定义控件是提升用户体验和界面个性化的重要手段。本文将通过浅显易懂的语言和实例,引导你了解自定义控件的基本概念、创建流程以及高级应用技巧,帮助你在开发过程中更好地掌握自定义控件的使用和优化。
85 10
|
4月前
|
数据可视化 Android开发 开发者
安卓应用开发中的自定义View组件
【10月更文挑战第5天】在安卓应用开发中,自定义View组件是提升用户交互体验的利器。本篇将深入探讨如何从零开始创建自定义View,包括设计理念、实现步骤以及性能优化技巧,帮助开发者打造流畅且富有创意的用户界面。
151 0
|
4月前
|
XML 前端开发 Java
安卓应用开发中的自定义View组件
【10月更文挑战第5天】自定义View是安卓应用开发的一块基石,它为开发者提供了无限的可能。通过掌握其原理和实现方法,可以创造出既美观又实用的用户界面。本文将引导你了解自定义View的创建过程,包括绘制技巧、事件处理以及性能优化等关键步骤。
|
4月前
|
缓存 搜索推荐 Android开发
安卓开发中的自定义控件实践
【10月更文挑战第4天】在安卓开发的海洋中,自定义控件是那片璀璨的星辰。它不仅让应用界面设计变得丰富多彩,还提升了用户体验。本文将带你探索自定义控件的核心概念、实现过程以及优化技巧,让你的应用在众多竞争者中脱颖而出。
|
4月前
|
前端开发 搜索推荐 Android开发
安卓开发中的自定义控件实践
【10月更文挑战第4天】在安卓开发的世界里,自定义控件如同画家的画笔,能够绘制出独一无二的界面。通过掌握自定义控件的绘制技巧,开发者可以突破系统提供的界面元素限制,创造出既符合品牌形象又提供卓越用户体验的应用。本文将引导你了解自定义控件的核心概念,并通过一个简单的例子展示如何实现一个基本的自定义控件,让你的安卓应用在视觉和交互上与众不同。

热门文章

最新文章

  • 1
    如何修复 Android 和 Windows 不支持视频编解码器的问题?
  • 2
    Android历史版本与APK文件结构
  • 3
    【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
  • 4
    【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
  • 5
    当flutter react native 等混开框架-并且用vscode-idea等编译器无法打包apk,打包安卓不成功怎么办-直接用android studio如何打包安卓apk -重要-优雅草卓伊凡
  • 6
    APP-国内主流安卓商店-应用市场-鸿蒙商店上架之必备前提·全国公安安全信息评估报告如何申请-需要安全评估报告的资料是哪些-优雅草卓伊凡全程操作
  • 7
    【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
  • 8
    Android经典面试题之Kotlin中Lambda表达式和匿名函数的区别
  • 9
    【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
  • 10
    【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
  • 1
    【01】噩梦终结flutter配安卓android鸿蒙harmonyOS 以及next调试环境配鸿蒙和ios真机调试环境-flutter项目安卓环境配置-gradle-agp-ndkVersion模拟器运行真机测试环境-本地环境搭建-如何快速搭建android本地运行环境-优雅草卓伊凡-很多人在这步就被难倒了
    13
  • 2
    Cellebrite UFED 4PC 7.71 (Windows) - Android 和 iOS 移动设备取证软件
    28
  • 3
    【03】仿站技术之python技术,看完学会再也不用去购买收费工具了-修改整体页面做好安卓下载发给客户-并且开始提交网站公安备案-作为APP下载落地页文娱产品一定要备案-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    34
  • 4
    Android历史版本与APK文件结构
    121
  • 5
    【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
    29
  • 6
    【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
    23
  • 7
    APP-国内主流安卓商店-应用市场-鸿蒙商店上架之必备前提·全国公安安全信息评估报告如何申请-需要安全评估报告的资料是哪些-优雅草卓伊凡全程操作
    60
  • 8
    【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
    37
  • 9
    当flutter react native 等混开框架-并且用vscode-idea等编译器无法打包apk,打包安卓不成功怎么办-直接用android studio如何打包安卓apk -重要-优雅草卓伊凡
    73
  • 10
    【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
    121