自定义View(1)

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了自定义View(1)相关的知识,希望对你有一定的参考价值。

参考技术A

经历过前面三篇啰啰嗦嗦的基础篇之后,终于到了进阶篇,正式进入解析自定义View的阶段。

至于本章节为什么要叫进阶篇(虽然讲的是基础的内容),因为从本篇开始,将会逐渐揭开自定义View的神秘面纱,每一篇都将比上一篇内容更加深入,利用所学的知识能够制作更加炫酷自定义View,就像在台阶上一样,每一篇都更上一层, 帮助大家一步步走向人生巅峰,出任CEO,赢取白富美。 误,是帮助大家更加了解那些炫酷的自定义View是如何制作的,达到举一反三的效果。

作为一个<b>有(hui)追(zhuang)求(B)</b>的程序员,肯定想做一些让人眼前一亮的程序效果,但是系统提供的那些一般很难满足,为了<b>梦(zhuang)想(B)</b>就必须要学习一些自定义View。下面我们就了解一些自定义View相关的东西。

自定义ViewGroup一般是利用现有的组件根据特定的布局方式来组成新的组件,大多继承自ViewGroup或各种Layout,包含有子View。

例如:一个应用内的底部导航条中的条目,一般都是上面为图标,下面是文字,那么这两个就可以用自定义ViewGroup组合成为一个Veiw,提供两个属性分别用来设置文字和图片即可,这样使用起来会方便很多。

在没有现成的View,需要自己实现的时候,就使用自定义View,一般继承自View,SurfaceView或其他的View,不包含子View。

例如:定义一个支持自动加载网络图片的ImageView,或制作一种特殊的动画效果。

<b>一般来说,自定义View在大多数情况下都有替代方案,利用图片或者组合动画来实现,但是使用后者可能会面临内存耗费过大,制作麻烦更诸多问题。</b>

View的构造函数有四种重载分别如下

可以看出,关于View构造函数的参数有多有少,先排除几个不常用的,留下常用的再研究。

<b>有四个参数的构造函数在API21的时候才添加上,我一般不使用,暂不考虑。</b>

有三个参数的构造函数中第三个参数是默认的Style,这里的默认的Style是指它在当前Application或Activity所用的Theme中的默认Style,且只有在明确调用的时候才会生效,以系统中的ImageButton为例说明:

<b>注意:即使你在View中使用了Style这个属性也不会调用三个参数的构造函数,所调用的依旧是两个参数的构造函数。</b>

<b>由于三个参数的构造函数第三个参数一般不用,暂不考虑,第三个参数的具体用法会在以后用到的时候详细介绍。</b>

排除了两个之后,只剩下一个参数和两个参数的构造函数,他们的详情如下:

以下方法调用的是<b>一个参数</b>的构造函数:

以下方法调用的是<b>两个参数</b>的构造函数:

关于构造函数先讲这么多,关于如何自定义属性和使用attrs中的内容,在后面会详细讲解,目前只需要知道这两个构造函数在何时调用即可。

========

测量View大小使用的是onMeasure函数,我们可以从这两个参数取出宽高的相关数据:

从上面可以看出 onMeasure 函数中有 widthMeasureSpec 和 heightMeasureSpec 这两个 int 类型的参数, 毫无疑问他们是和宽高相关的, <b>但它们其实不是宽和高, 而是由宽、高和各自方向上对应的模式来合成的一个值:</b>

在int类型的32位二进制位中,31-30这两位表示模式,29~0这三十位表示宽和高的实际值。

以数值1080(二进制为: 1111011000)为例(其中模式和实际数值是连在一起的,为了展示我将他们分开了):

实际上关于上面的东西了解即可,在实际运用之中只需要记住有三种模式,用 MeasureSpec 的 getSize是获取数值, getMode是获取模式即可。

如果对View的宽高进行修改了,<b>不要调用super.onMeasure(widthMeasureSpec,heightMeasureSpec);</b>
要调用<b>setMeasuredDimension(widthsize,heightsize);</b> 这个函数。

======

这个函数在视图大小发生改变时调用:

onSizeChanged如下:

可以看出,它又四个参数,分别为 宽度,高度,上一次宽度,上一次高度。

这个函数比较简单,我们只需关注 宽度(w), 高度(h) 即可,这两个参数就是View最终的大小。

=========

<b>确定布局的函数是onLayout,它用于确定子View的位置,在自定义ViewGroup中会用到,他调用的是子View的layout函数。</b>

不过关于View的layout函数我们一般无需关注,因为在一般情况下我们只需关注View自身的坐标系即可,除非View状态与在父VIew所处位置相关。

在自定义ViewGroup中,onLayout一般是循环取出子View,然后经过计算得出各个子View位置的坐标值,然后用以下函数设置子View位置。

四个参数分别为:

具体可以参考 坐标系 这篇文章:

PS:关于onLayout这个函数在讲解自定义ViewGroup的时候会详细讲解。

========

onDraw是实际绘制的部分,也就是我们真正关心的部分,使用的是Canvas绘图。

关于Canvas绘图另分一章吧,本来想写一些关于Canvas基本操作的的,可是篇幅太长了QAQ, 留个尾巴下一篇再写吧,毕竟Canvas绘图也是一个比较庞大的东西,也不是三言两语就能讲明白的,就到这里吧。

======

自定义完View之后,一般会对外暴露一些接口,用于操作View的相关属性,控制View的状态等,或者需要监听View的变化,具体还是稍后再讲吧(继续挖坑)。

PS :实际上ViewGroup是View的一个子类。

View
ViewGroup
View.MeasureSpec
onMeasure,MeasureSpec源码 流程 思路详解
android中自定义样式与View的构造函数中的第三个参数defStyle的意义
android view构造函数研究
Android View构造方法第三参数使用方法详解
Android 自定义View onMeasure方法的实现
Android API指南(二)自定义控件02之 onMeasure
Android中View的绘制过程 onMeasure方法简述

自定义View系列一 自定义View的构造函数,自定义属性

转载请标明出处:
http://blog.csdn.net/zq2114522/article/details/51147893
本文出自:【梁大盛的博客】

自定义View系列一 自定义View的构造函数,自定义属性

引:自定义View对于Android开发者是一道坎.虽然说是坎但是也得走过去的!此系列文章作为学习自定义View的一系列学习笔记.

在进入学习自定义View的殿堂,第一件需要弄清楚的事情是该怎么定义自定义View的构造函数.因为这里就是自定义View的入口.

哪里调用自定义View的构造函数?

会调用自定义View的构造函数莫非就只有两个地方.
第一个是通过在代码创建一个自定义的View.

如:

View view = new MyView(this);

第二个就是通过在布局文件使用自定义View,当然这种情况系统会自动调用我们的构造函数.

如:

<?xml version="1.0" encoding="utf-8"?>
<com.example.dsliang.viewdemo.MyView 
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"/>
这两种情况的区别:

    在代码里面创建MyView对象此时是调用只有个参数的构造函数,在布局文件里面使用MyView这情况系统默认是调用两个参数的构造函数.然而区别就是这样了.为什么通过布局文件使用MyView的时候就会使用两个参数的构造函数呢?我们理所当然的知道在布局文件里面使用一个View的时候会添加各种属性.像layout_width,layout_height等等.然而在创建View的时候就需要获取这些属性吧?事实上就是通过参数传递进去的.所以当在布局文件里面使用View,为啥调用两个参数构造函数就是这原因了!

接下来就看看构造函数的面貌.

该定义什么构造函数?

构造函数本应该不复杂,但是要完全弄清楚就真的就变得有点复杂了.从简单原则出发,我们目标是能把事情说清楚把概念讲明白即可.

一开始View的构造函数就只有三个发展到现在在Api 21引入了4个参数的构造函数(事实上我也没弄懂那是什么玩意).


(前面三个构造函数在Api 1引入,最后一个构造函数(4个参数的)在Api 21才引入)

着重说前面两个构造函数.(方面后面阐述,只有一个参数的构造函数叫C1,两个参数的叫C2如此类推)
前面讨论通过在代码里面创建MyView就是调用C1构造函数.布局使用MyView就是调用C2构造函数.
理解起来也是很简单的.看看最简单的自定义MyView代码吧.

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>

<com.example.dsliang.viewdemo.MyView xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:background="#FFFFFF00"/>

MyView.java

package com.example.dsliang.viewdemo;

import android.content.Context;
import android.util.AttributeSet;
import android.util.Log;
import android.view.View;

/**
 * Created by dsliang on 2016/4/8.
 */
public class MyView extends View 

    static final String TAG = MyView.class.getSimpleName();

    public MyView(Context context) 
        super(context);
        Log.d(TAG, "ViewDemo(1)");
    

    public MyView(Context context, AttributeSet attrs) 
        super(context, attrs);
        Log.d(TAG, "ViewDemo(2)");
    


MainActivity.java

import android.support.v7.app.AppCompatActivity;

public class MainActivity extends AppCompatActivity 

    private static final String TAG = MainActivity.class.getSimpleName();

    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    

效果图:

(Logcat可以看到是调用了C2)

就简单几句代码已经实现了自定义View了!是不是觉得自定义View也不就那样了?接下来是本章的最后一个内容,怎么自定义属性.

怎么自定义属性

我们自定义View在很多时候都需要自定义一些属于此View的一些属性.用一个简单笼统的理解.怎么把布局文件里面使用的属性传递到代码里面.这就是自定义属性所使用的场景.

在这里提出一个场景:在布局文件使用我们的自定义View.并且通过debug_text属性传递一句话,在自定义View里面获取debug_text并且通过Logcat输出相应的文本.

定义属性

在布局文件里面不是想使用debug_text就能使用,必须先定义以后才能使用.
定义属性步骤:
1.在values包下新建attrs.xml文件
2.定义相应的属性
如:

attrs.xml

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="MyView">
        <attr name="debug_text" format="string" />
    </declare-styleable>
</resources>

接下来我们就可以在自定义View里面使用我们已经定义的属性.

使用属性

已经定义属性以后使用属性是很简单的事情.
1.添加xml命名空间
2.通过相应的命名空间使用属性

<?xml version="1.0" encoding="utf-8"?>

<com.example.dsliang.viewdemo.MyView xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto/com.example.dsliang.viewdemo"
    <!--app 是可以随意命名的-->
    <!--http://schemas.android.com/apk/res-auto/com.example.dsliang.viewdemo 格式是固定的-->
    <!--http://schemas.android.com/apk/res-auto/ 加包名-->
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:background="#FFFFFF00"
    <!-- 引用自定义属性 -->
    app:debug_text="hello,,MyView"
    />

获取属性

到最后一步了.这里是介绍怎么在MyView代码里面获取在布局文件定义的属性.不知道你激动不激动,反正我现在是很激动!

首先必须知道我们必须在C2里面获取布局文件传递进来的属性.
这里就通过贴出代码并且在代码里面添加注解的方式解说.

MyView.java部分代码

public MyView(Context context, AttributeSet attrs) 
        super(context, attrs);
        Log.d(TAG, "ViewDemo(2)");

        //通过attrs,R.styleable.MyView获取一个TypedArray实实例
        //TypedArray实例里面已经存放好我们需要或许的属性了
        TypedArray a = context.obtainStyledAttributes(attrs, R.styleable.MyView);
        //使用R.styleable.MyView_debug_text索引我们需要的属性
        String debugText = a.getString(R.styleable.MyView_debug_text);
        //释放TypedArray实例,使用完必须释放
        a.recycle();

        Log.d(TAG, "debug_text: " + debugText);
    

效果图:

总结:
到此自定义View的构造函数和自定义属性基本用法都介绍了一遍.当然是本着点到即止的宗旨出发.然后发现一个细节吗?我么MyView的witdh和height属性均为wrap_content,但是Demo确实填充整个父布局了.当然这一个问题就是下一篇要详尽介绍的onMeasure方法了.

注意:例子都是很简单的例子,只是希望能把问题说清楚.

参考


attr属性 http://blog.csdn.net/dalancon/article/details/9701855
View http://android.xsoftlab.net/reference/android/view/View.html

以上是关于自定义View(1)的主要内容,如果未能解决你的问题,请参考以下文章

Android 自定义View

Android 自定义View

自定义控件2.第一个自定义view

自定义View系列一 自定义View的构造函数,自定义属性

自定义 view - 自定义属性

Android自定义View