【Android自定义View实战】之你应该明白的事儿

2024-05-22 06:58

本文主要是介绍【Android自定义View实战】之你应该明白的事儿,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

转载请注明出处:http://blog.csdn.net/linglongxin24/article/details/52910443【DylanAndroid的csdn博客】


在Android的实际开发中,我们Android系统本身已经给我们提供了很丰富的UI以及各种实用的控件,例如TextView,Button,ImageView等。用这些基础控件已经能够实现非常优美的界面以及功能。然而在实际的开发中,我们由于客户的各种需求,App开发的各种标新立异,追求个性化,所以,导致我们用这些最基础的控件已经不能够满足我们的各种个性化需求。那么,我们就要考虑去自定义控件来完成我们的要求,要真正去完成一个自定义控件,我们先来看看这个控件的继承体系:

1.View体系

Android 应用中的所有用户界面元素都是使用 View 和 ViewGroup 对象构建而成。View 对象用于在屏幕上绘制可供用户交互的内容。ViewGroup 对象用于储存其他 View(和 ViewGroup)对象,以便定义界面的局部。

Android 提供了一系列 View 和 ViewGroup 子类,可为您提供常用输入控件(如按钮和文本字段)和各种布局模式(如线性布局或相对布局)。

这里写图片描述

2.View的关键绘制流程和关键的生命周期

Created with Raphaël 2.1.0 Constructor------>构造函数 onMeasure------>测量View大小 onSizeChanged------>确定View大小 onLayout------>确定View布局的位置 onDraw------>绘制View中的内容 视图状态是否改变 显示 invalidate yes no

综上所述:View的关键生命周期为:
构造View() –> onMeasure() –> onSizeChanged() –> onLayout() –> onDraw()

3.构造函数的自定义属性

  • 构造函数

    public CustomView(Context context) {super(context);
    }public CustomView(Context context, AttributeSet attrs) {super(context, attrs);
    }public CustomView(Context context, AttributeSet attrs, int defStyleAttr) {super(context, attrs, defStyleAttr);
    }
    • 首先在res的values文件夹下面建一个attrs.xml文件,如下图所示:
      这里写图片描述
    • 然后在里面定义如下内容
    <?xml version="1.0" encoding="utf-8"?>
    <resources>
    <declare-styleable name="CircleImageView"><attr name="border_width" format="dimension" /><attr name="border_color" format="color" />
    </declare-styleable>
    </resources>
  • 首先应该在布局文件的跟布局当中定义命名空间
xmlns:app="http://schemas.android.com/apk/res-auto"
  • 在xml中的用法:如下所示
 <com.bm.dylan.view.CircleImageView
                android:id="@+id/civ_personal_information_head"android:layout_width="50dp"android:layout_height="50dp"android:layout_alignParentRight="true"android:layout_centerVertical="true"android:src="@mipmap/personalcenter_head"app:border_color="@color/white" />app:border_width="2dp" />
  • 最后在构造函数中这样获取:

    
    public CircleImageView(Context context, AttributeSet attrs, int defStyle) {super(context, attrs, defStyle);super.setScaleType(SCALE_TYPE);TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.CircleImageView, defStyle, 0);int mBorderWidth = a.getDimensionPixelSize(R.styleable.CircleImageView_border_width, DEFAULT_BORDER_WIDTH);int  mBorderColor = a.getColor(R.styleable.CircleImageView_border_color, DEFAULT_BORDER_COLOR);a.recycle();}
  • 在xml中可以定义的属性如下

编号类型format的值在xml中定义值示例
1reference参考指定Theme中资源IDreferenceapp:name=”@string/label”
2color:颜色colorapp:Textcolor=”#ff0000”
3boolean:布尔值booleanapp:isVisible=”false”
4dimension:尺寸值dimensionapp:myWidth=”100dp”
5float:浮点型floatapp:fromAlpha=”0.3”
6integer:整型integerapp:framesCount=”22”
7string:字符串stringapp:name=”My name is yuan dong liang”
8fraction:百分数fractionapp:pivotX=”200%”
9enum:枚举enumapp:borderRadius=”circle”
10flag:位或运算fractionandroid:windowSoftInputMode=”stateUnspecified
11多种符合类型colorapp:background = “@drawable/图片ID|#00FF00”|

4..测量View大小(onMeasure)

  • 代码
    @Overrideprotected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {// TODO Auto-generated method stub  super.onMeasure(widthMeasureSpec, heightMeasureSpec);  System.out.println(widthMeasureSpec+":"+heightMeasureSpec);  // 如果是自定ViewGroup,计算自定义的ViewGroup中所有子控件的大小  //measureChildren(widthMeasureSpec, heightMeasureSpec);  int w = getMeasureWidth(widthMeasureSpec);    int h = getMeasureHeight(heightMeasureSpec); setMeasuredDimension(w, h);  //必须调用此方法,否则会抛出异常  }private int getMeasureHeight(int heightMeasureSpec) {  int result = 0;  int size = MeasureSpec.getSize(heightMeasureSpec);  //每次调用此方法,测量用到的size会发生变化  int mode = MeasureSpec.getMode(heightMeasureSpec);  //根据定义的Layout_width,Layout_height,会对此值产生影响  if (mode == MeasureSpec.EXACTLY) {  result = size;  } else if (mode == MeasureSpec.UNSPECIFIED) {  result = getPaddingLeft()  + getPaddingRight();  } else {  result = Math.min(result, size);  }  System.out.println("Height size:" + size);    System.out.println("Height mode:" + mode);  return result;  
}  private int getMeasureWidth(int widthMeasureSpec) {  int result = 0;  int size = MeasureSpec.getSize(widthMeasureSpec);  int mode = MeasureSpec.getMode(widthMeasureSpec);  if (mode == MeasureSpec.EXACTLY) {  result = size;  } else if (mode == MeasureSpec.UNSPECIFIED) {  result =getPaddingTop()  + getPaddingBottom();  } else {  result = Math.min(result, size);  }  System.out.println("Width size:" + size);  System.out.println("Width mode:" + mode);  return result;  
}  
  • MeasureSpec

    从上面可以看出 onMeasure 函数中有 widthMeasureSpec 和 heightMeasureSpec 这两个 int 类型的参数, 毫无疑问他们是和宽高相关的, 但它们其实不是宽和高, 而是由宽、高和各自方向上对应的测量模式来合成的一个值:对于详细测量值( measureSpec )需要两样东西来确定它,那就是大小(size)和模式(mode)。 而 measureSpec,size,mode他们三个的关系,都封装在View类中的一个内部类里,名叫 MeasureSpec 。
    测量模式一共有三种, 被定义在 Android 中的 View 类的一个内部类View.MeasureSpec中:

模式二进制数值描述
UNSPECIFIED00默认值,父控件没有给子view任何限制,子View可以设置为任意大小。
EXACTLY01表示父控件已经确切的指定了子View的大小。
AT_MOST10表示子View具体大小没有尺寸限制,但是存在上限,上限一般为父View大小。

我们重写onMeasure()所要实现的最终目的。它的作用就是存储我们测量好的宽高值。

5.确定View大小(onSizeChanged)

因为View的大小不仅由View本身控制,而且受父控件的影响,所以我们在确定View大小的时候最好使用系统提供的onSizeChanged回调函数。它又四个参数,分别为 宽度,高度,上一次宽度,上一次高度。这个函数比较简单,我们只需关注 宽度(w), 高度(h) 即可,这两个参数就是View最终的大小。

 @Overrideprotected void onSizeChanged(int w, int h, int oldw, int oldh) {super.onSizeChanged(w, h, oldw, oldh);}

6.确定子View布局位置(onLayout)

确定布局的函数是onLayout,它用于确定子View的位置,在自定义ViewGroup中会用到,他调用的是子View的layout函数。在自定义ViewGroup中,onLayout一般是循环取出子View,然后经过计算得出各个子View位置的坐标值,然后用以下函数设置子View位置。

 @Override  protected void onLayout(boolean changed, int l, int t, int r, int b) {  // 记录总高度  int mTotalHeight = 0;  // 遍历所有子视图  int childCount = getChildCount();  for (int i = 0; i < childCount; i++) {  View childView = getChildAt(i);  // 获取在onMeasure中计算的视图尺寸  int measureHeight = childView.getMeasuredHeight();  int measuredWidth = childView.getMeasuredWidth();  childView.layout(l, mTotalHeight, measuredWidth, mTotalHeight  + measureHeight);  mTotalHeight += measureHeight;  }  }  
四个参数分别为:
名称说明对应的函数
lView左侧距父View左侧的距离getLeft();
tView顶部距父View顶部的距离getTop();
rView右侧距父View左侧的距离getRight();
bView底部距父View顶部的距离getBottom();

7.绘制内容(onDraw)

  @Overrideprotected void onDraw(Canvas canvas) {super.onDraw(canvas);//利用Canvas 画图}

8.绘图知识

  • Canvas(画布)类

    画笔属性设置好之后,还需要将图像绘制到画布上。Canvas类可以用来实现各种图形的绘制工作,如绘制直线、矩形、圆等等。Canvas绘制常用图形的方法如下:
    绘制直线:canvas.drawLine(float startX, float startY, float stopX, float stopY, Paint paint);

      绘制矩形:canvas.drawRect(float left, float top, float right, float bottom, Paint paint);

      绘制圆形:canvas.drawCircle(float cx, float cy, float radius, Paint paint);

      绘制字符:canvas.drawText(String text, float x, float y, Paint paint);

      绘制图形:canvas.drawBirmap(Bitmap bitmap, float left, float top, Paint paint);

  • Paint(画笔)类
    要绘制图形,首先得调整画笔,按照自己的开发需要设置画笔的相关属性。Pain类的常用属性设置方法如下:
 setAntiAlias();             //设置画笔的锯齿效果setColor();                 //设置画笔的颜色setARGB();                  //设置画笔的A、R、G、B值setAlpha();                 //设置画笔的Alpha值setTextSize();              //设置字体的尺寸setStyle();                  //设置画笔的风格(空心或实心)setStrokeWidth();            //设置空心边框的宽度getColor();                  //获取画笔的颜色

9.对外提供操作方法和监听回调

自定义完View之后,一般会对外暴露一些接口,用于控制View的状态等,或者监听View的变化.也就是定义接口。我们的像Button的点击事件就是这样的原理。

10.自定义View的分类

  • 在原有的控件基础上扩展,比如继承ImageView去实现圆形头像等。
  • 通过组合控件来自定义View,比如标题栏就可以通过组合控件来实现,左边一个返回按钮,中间一个显示文本,右边可有可无的提交按钮。
  • 完全重写onDraw()自定义View。

这篇关于【Android自定义View实战】之你应该明白的事儿的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/996001

相关文章

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

使用 sql-research-assistant进行 SQL 数据库研究的实战指南(代码实现演示)

《使用sql-research-assistant进行SQL数据库研究的实战指南(代码实现演示)》本文介绍了sql-research-assistant工具,该工具基于LangChain框架,集... 目录技术背景介绍核心原理解析代码实现演示安装和配置项目集成LangSmith 配置(可选)启动服务应用场景

Android里面的Service种类以及启动方式

《Android里面的Service种类以及启动方式》Android中的Service分为前台服务和后台服务,前台服务需要亮身份牌并显示通知,后台服务则有启动方式选择,包括startService和b... 目录一句话总结:一、Service 的两种类型:1. 前台服务(必须亮身份牌)2. 后台服务(偷偷干

在Java中使用ModelMapper简化Shapefile属性转JavaBean实战过程

《在Java中使用ModelMapper简化Shapefile属性转JavaBean实战过程》本文介绍了在Java中使用ModelMapper库简化Shapefile属性转JavaBean的过程,对比... 目录前言一、原始的处理办法1、使用Set方法来转换2、使用构造方法转换二、基于ModelMapper

Java实战之自助进行多张图片合成拼接

《Java实战之自助进行多张图片合成拼接》在当今数字化时代,图像处理技术在各个领域都发挥着至关重要的作用,本文为大家详细介绍了如何使用Java实现多张图片合成拼接,需要的可以了解下... 目录前言一、图片合成需求描述二、图片合成设计与实现1、编程语言2、基础数据准备3、图片合成流程4、图片合成实现三、总结前

CSS自定义浏览器滚动条样式完整代码

《CSS自定义浏览器滚动条样式完整代码》:本文主要介绍了如何使用CSS自定义浏览器滚动条的样式,包括隐藏滚动条的角落、设置滚动条的基本样式、轨道样式和滑块样式,并提供了完整的CSS代码示例,通过这些技巧,你可以为你的网站添加个性化的滚动条样式,从而提升用户体验,详细内容请阅读本文,希望能对你有所帮助...

nginx-rtmp-module构建流媒体直播服务器实战指南

《nginx-rtmp-module构建流媒体直播服务器实战指南》本文主要介绍了nginx-rtmp-module构建流媒体直播服务器实战指南,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有... 目录1. RTMP协议介绍与应用RTMP协议的原理RTMP协议的应用RTMP与现代流媒体技术的关系2

Android kotlin语言实现删除文件的解决方案

《Androidkotlin语言实现删除文件的解决方案》:本文主要介绍Androidkotlin语言实现删除文件的解决方案,在项目开发过程中,尤其是需要跨平台协作的项目,那么删除用户指定的文件的... 目录一、前言二、适用环境三、模板内容1.权限申请2.Activity中的模板一、前言在项目开发过程中,尤

C语言小项目实战之通讯录功能

《C语言小项目实战之通讯录功能》:本文主要介绍如何设计和实现一个简单的通讯录管理系统,包括联系人信息的存储、增加、删除、查找、修改和排序等功能,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录功能介绍:添加联系人模块显示联系人模块删除联系人模块查找联系人模块修改联系人模块排序联系人模块源代码如下

Golang操作DuckDB实战案例分享

《Golang操作DuckDB实战案例分享》DuckDB是一个嵌入式SQL数据库引擎,它与众所周知的SQLite非常相似,但它是为olap风格的工作负载设计的,DuckDB支持各种数据类型和SQL特性... 目录DuckDB的主要优点环境准备初始化表和数据查询单行或多行错误处理和事务完整代码最后总结Duck