為什么要自定義View洒沦?
Android系統(tǒng)提供了一系列的原生控件赁温,但這些原生控件并不能夠滿足我們的需求時(shí),我們就需要自定義View了墅拭。
自定義View流程
一般來(lái)說(shuō)活玲,自定義view要重寫(xiě)onMeasure()以及onDraw()這兩個(gè)方法。看方法名字就知道舒憾,onMeasure()是負(fù)責(zé)測(cè)量控件的大小镀钓,onDraw()方法是負(fù)責(zé)將控件畫(huà)出來(lái)。
onMeasure方法
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
在這個(gè)方法中珍剑,一共有兩個(gè)參數(shù)掸宛。這兩個(gè)參數(shù)分別表示控件寬高的大小以及測(cè)量模式,兩個(gè)都是32位的int型數(shù)據(jù)招拙,其中前面2位表示測(cè)量模式唧瘾,后兩位表示控件的大小。谷歌已經(jīng)封裝好方法給我們别凤,可以直接通過(guò)方法即可獲取測(cè)量模式以及大小饰序。
//獲取測(cè)量模式
int mode = MeasureSpec.getMode(widthMeasureSpec);
//獲取大小
int size = MeasureSpec.getSize(widthMeasureSpec);
測(cè)量模式一種有3種
測(cè)量模式 | 含義 |
---|---|
EXACTLY | 父容器檢測(cè)出view精確的大小,view最終大小為測(cè)量的大小 |
AT_MOST | 父容器指定了view的最大值规哪,view的大小不可以超過(guò)這個(gè)最大值 |
UNSPECIFIED | 父容器對(duì)view沒(méi)有限制求豫,view本身想要多大就多大 |
測(cè)量模式跟我們平時(shí)開(kāi)發(fā)在布局文件中寫(xiě)的match_parent、wrap_content有什么關(guān)系呢诉稍?
match_parent對(duì)應(yīng)的是EXACTLY.我們?cè)趺蠢斫饽仳鸺危科鋵?shí)很簡(jiǎn)單,我們布局里面寫(xiě)match_parent表示的是撐滿父布局的剩余空間杯巨。父布局的剩余空間是確定的蚤告,因此是view的大小是一個(gè)確定的值,所以是EXACTLY.
wrap_content對(duì)應(yīng)的是AT_MOST.我們?cè)诓季种惺褂脀rap_content的意思是包裹控件的內(nèi)容服爷。但是這個(gè)時(shí)候父控件大小是不確定的杜恰,自然子view占用的大小也是不確定的。但是怎么理解父布局給出的建議大小呢仍源?事實(shí)上心褐,父布局給出的建議大小其實(shí)就是父布局可以獲取的最大的大小。具體為什么笼踩,可以參考View的繪制流程一文逗爹,在這里不討論。
關(guān)于測(cè)量模式可以參考此文:Android View的繪制流程
重寫(xiě)onMeasure方法
我們理論說(shuō)得再多還不如動(dòng)手實(shí)踐一次嚎于,那就直接Show The Code No BB.
我們簡(jiǎn)單繪制一個(gè)圓形的view掘而。
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
int width = getViewSize(100, widthMeasureSpec);
int height = getViewSize(100, heightMeasureSpec);
if (width < height) {
height = width;
} else {
width = height;
}
setMeasuredDimension(width, height);
}
private int getViewSize(int defaultSize, int measureSpec) {
int resultSize = defaultSize;
int size = MeasureSpec.getSize(measureSpec);
int mode = MeasureSpec.getMode(measureSpec);
switch (mode) {
case MeasureSpec.EXACTLY:
resultSize = size;
break;
case MeasureSpec.AT_MOST:
resultSize = size;
break;
case MeasureSpec.UNSPECIFIED:
resultSize = defaultSize;
break;
}
return resultSize;
}
布局如下:
<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity">
<com.example.mazhi.customview.MyView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:view_color="@color/colorAccent" />
</android.support.constraint.ConstraintLayout>
在布局中,我們view是wrap_content的匾旭,因此父布局給我們的測(cè)量模式一定是AT_MOST,size一定就是屏幕的寬度圃郊。我們?cè)谶@里要實(shí)現(xiàn)的是一個(gè)正方形价涝,因此width以及height設(shè)置了相等。其他的就不多解釋了持舆,大家一看就懂色瘩。
重寫(xiě)onDraw方法
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
//半徑
int r = getMeasuredHeight() / 2;
canvas.drawCircle(r, r, r, paint);
}
算出半徑以及圓心伪窖,直接調(diào)用api即可,沒(méi)什么難的居兆,也不多解釋了吧覆山。
總結(jié)一下,簡(jiǎn)單的自定義view可以直接重寫(xiě)onMeasure方法以及onDraw方法即可泥栖。onMeasure方法負(fù)責(zé)測(cè)量view的大小簇宽,onDraw方法復(fù)制繪制。