自定義組件:前置

一、概述

Android SDK為我們提供了一套完整的組件庫,當這些原生的組件滿足不了我們的需求,就需要自定義組件了.

學(xué)習(xí)自定義組件最好的方式是閱讀源碼,最權(quán)威的自然是Google提供的API Demos,這里面包含了開發(fā)的各種知識,然后就是Githup上優(yōu)秀的開源項目,這些開源項目會給我們提供不同的思路,不管是簡單的組件還是復(fù)雜的組件,都是有規(guī)律的,通過閱讀優(yōu)秀的源碼,了解別人的思維模式和編程技巧,有時候你會豁然開朗.

二、分類

1、View分類

View類是Android中所有組件的基類,也就是老祖宗,其它的組件都是View的子孫,View類還有一個重要的子類:ViewGroup類,ViewGroup通常作為其它組件的容器使用.

Android中所有UI組件都是建立在View、ViewGroup基礎(chǔ)之上,對于一個Android應(yīng)用的用戶界面來說:ViewGroup作為容器來盛放其它組件,ViewGroup里面除了可以盛放普通View組件之外,還可以再次盛放ViewGroup,下圖顯示一個用戶界面的組件層次.

2、自定義組件分類

一般來說,有4種方式用來實現(xiàn)自定義組件:

(1)繼承View類

這種方式基本是從零開始自定義了,工作量比較大

(2)繼承View子類(非ViewGroup)

比如繼承TextView、ImageView

(3)繼承ViewGroup類

這里ViewGroup包括ViewGroup的子類,像我們最常用的LinearLayout、RelativeLayout

(4)自定義組合控件

將多個已有組件組合形成一個新的組件

三、構(gòu)造方法

public class XView extends View {

    /**
     * 含1個參數(shù)的構(gòu)造方法
     */
    public XView(Context context) {
        super(context);
    }

    /**
     * 含2個參數(shù)的構(gòu)造方法
     */
    public XView(Context context, AttributeSet attrs) {
        super(context, attrs);
    }

    /**
     * 含3個參數(shù)的構(gòu)造方法
     */
    public XView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

    /**
     * 含4個參數(shù)的構(gòu)造方法(API>=21)
     */
    public XView(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
        super(context, attrs, defStyleAttr, defStyleRes);
    }
}

學(xué)習(xí)一個類是從構(gòu)造方法開始,自定義組件也是如此.

自定義組件一共4個構(gòu)造方法:
1、第一個構(gòu)造方法:只有1個參數(shù)context,在代碼中創(chuàng)建組件時會調(diào)用該構(gòu)造,比如在Activity中:

XView  xView = new XView(this);

2、第二個構(gòu)造方法:有2個參數(shù),在layout布局文件中使用時會調(diào)用該構(gòu)造,比如:

    <com.fgq.demo.XView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />

PS:在布局文件中使用自定義組件的寫法:完整包名+類名

參數(shù)attrs表示當前配置的屬性集,在布局文件中定義的寬、高等屬性都包含在attrs中.

3、第三個構(gòu)造方法:有3個參數(shù),這個構(gòu)造方法系統(tǒng)不會調(diào)用,一般在有自定義屬性時,我們會在第二個構(gòu)造方法中手動調(diào)用該構(gòu)造.

4、第四個構(gòu)造方法:這個構(gòu)造在API>=21(5.0)才能使用,除非你的minSdkVersion>=21,不然不會用到.

所以自定義組件構(gòu)造方法最常見的寫法像下面這樣:

    public XView(Context context) {
        this(context, null);
    }

    public XView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public XView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
    }

四、onMeasure、onDraw

在了解完構(gòu)造方法之后就要介紹兩個方法:onMeasure、onDraw,自定義View大部分時候需要重寫這兩個方法:

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
    }

1、onMeasure

onMeasure方法用來測量組件的大小,在Android中,組件的大小都是通過該方法進行測量,不管界面多么復(fù)雜,每個組件都負責(zé)計算自己的大小,如果不重寫該方法,父類View會有默認的實現(xiàn).

那么為什么要重寫該方法?我們在xml布局文件中不是設(shè)置了寬、高了嗎?

在初學(xué)Android時,我們就知道:在xml布局文件中,layout_width、layout_height是可以不用寫具體尺寸的,比如寬高都設(shè)置為wrap_content(包裹內(nèi)容),此時并沒有指定組件的真正大小,但是繪制到屏幕上的View必須要有具體的寬高,所以我們需要重寫onMeasure自己去測量寬高.

當然,對于測量組件大小,父類View有默認的處理,但是如果默認處理滿足不了我們的需求,此時就需要重寫onMeasure方法了.

2、onDraw

這個方法我們應(yīng)該很熟悉了,在Graphics2D API一章中我們一直在重寫該方法,該方法用來繪制組件的內(nèi)容.

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,763評論 6 539
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,238評論 3 428
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,823評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,604評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,339評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,713評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,712評論 3 445
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,893評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,448評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 41,201評論 3 357
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,397評論 1 372
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,944評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,631評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,033評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,321評論 1 293
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,128評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,347評論 2 377