Android自定義View

為什么要自定義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ù)制繪制。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末吧享,一起剝皮案震驚了整個(gè)濱河市魏割,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌钢颂,老刑警劉巖钞它,帶你破解...
    沈念sama閱讀 218,607評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異殊鞭,居然都是意外死亡遭垛,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,239評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)操灿,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)锯仪,“玉大人,你說(shuō)我怎么就攤上這事牲尺÷牙遥” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,960評(píng)論 0 355
  • 文/不壞的土叔 我叫張陵谤碳,是天一觀的道長(zhǎng)溃卡。 經(jīng)常有香客問(wèn)我,道長(zhǎng)蜒简,這世上最難降的妖魔是什么瘸羡? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,750評(píng)論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮搓茬,結(jié)果婚禮上犹赖,老公的妹妹穿的比我還像新娘。我一直安慰自己卷仑,他們只是感情好峻村,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,764評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著锡凝,像睡著了一般粘昨。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,604評(píng)論 1 305
  • 那天张肾,我揣著相機(jī)與錄音芭析,去河邊找鬼。 笑死吞瞪,一個(gè)胖子當(dāng)著我的面吹牛馁启,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播芍秆,決...
    沈念sama閱讀 40,347評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼惯疙,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了浪听?” 一聲冷哼從身側(cè)響起螟碎,我...
    開(kāi)封第一講書(shū)人閱讀 39,253評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎迹栓,沒(méi)想到半個(gè)月后掉分,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,702評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡克伊,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,893評(píng)論 3 336
  • 正文 我和宋清朗相戀三年酥郭,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片愿吹。...
    茶點(diǎn)故事閱讀 40,015評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡不从,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出犁跪,到底是詐尸還是另有隱情椿息,我是刑警寧澤,帶...
    沈念sama閱讀 35,734評(píng)論 5 346
  • 正文 年R本政府宣布坷衍,位于F島的核電站寝优,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏枫耳。R本人自食惡果不足惜乏矾,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,352評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望迁杨。 院中可真熱鬧钻心,春花似錦、人聲如沸铅协。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,934評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)狐史。三九已至痒给,卻和暖如春坯钦,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背侈玄。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,052評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留吟温,地道東北人序仙。 一個(gè)月前我還...
    沈念sama閱讀 48,216評(píng)論 3 371
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像鲁豪,于是被迫代替她去往敵國(guó)和親潘悼。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,969評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容