中文字幕在线观看,亚洲а∨天堂久久精品9966,亚洲成a人片在线观看你懂的,亚洲av成人片无码网站,亚洲国产精品无码久久久五月天

Android自定義控件常用方法總結(jié)

2018-07-20    來源:編程學(xué)習(xí)網(wǎng)

容器云強(qiáng)勢上線!快速搭建集群,上萬Linux鏡像隨意使用

inflate

inflate方法常常用來解析一個(gè)xml布局文件,在自定義組合式控件中常常使用,使用的姿勢包括:

View.inflate(context, resource, root)
LayoutInflater.from(getContext()).inflate(resource, root);

而View.inflate其實(shí)還是調(diào)用的LayoutInflater去解析一個(gè)xml:

public static View inflate(Context context, int resource, ViewGroup root) {
        LayoutInflater factory = LayoutInflater.from(context);
        return factory.inflate(resource, root);
    }

所以這兩種姿勢沒啥區(qū)別,這里來討論一下inflate(resouce, root)的返回值,參數(shù)resource是布局資源,root是傳入的一個(gè)根節(jié)點(diǎn)。如果root傳入一個(gè)null,inflate就會(huì)解析resource對應(yīng)的xml,返回這個(gè)xml中的根節(jié)點(diǎn),如果root傳入不為null,inflate會(huì)解析這個(gè)xml布局并且添加到根節(jié)點(diǎn)root下,然后返回根節(jié)點(diǎn)root。

另外還有帶三個(gè)參數(shù)的inflate方法:

inflate(int resource, ViewGroup root, boolean attachToRoot)

這里多了一個(gè)參數(shù)attachToRoot,如果root為null,則返回解析后的xml布局中的根節(jié)點(diǎn);如果root不為null,attachToRoot為true,inflate會(huì)解析這個(gè)xml布局并且添加到根節(jié)點(diǎn)root下,然后返回根節(jié)點(diǎn)root;如果root不為null,attachToRoot為false,inflate會(huì)解析這個(gè)xml布局但不會(huì)添加到根節(jié)點(diǎn)root下,然后返回解析后的xml布局中的根節(jié)點(diǎn),這時(shí)候root的作用只是為xml中的根節(jié)點(diǎn)提供布局參數(shù)的屬性,因?yàn)閤ml中的根節(jié)點(diǎn)不知道自己的父容器是誰,所以如果沒有人給它提供的話,它的布局參數(shù)就會(huì)失效。

onFinishInflate

onFinishInflate是當(dāng)所有的孩子都解析完后的一個(gè)調(diào)用。比如我們自定義一個(gè)ViewGroup,想要去找到孩子做一些設(shè)置,這時(shí)候如果在自定義ViewGroup的構(gòu)造函數(shù)去findViewById的話,會(huì)返回一個(gè)null,因?yàn)榇藭r(shí)孩子還沒有解析好,也就是還沒有生出來。這時(shí)候我們可以去覆寫onFinishInflate,當(dāng)孩子解析好后再去find。

requestLayout

關(guān)于requestLayout的介紹比較多,requestLayout()方法會(huì)觸發(fā)measure過程和layout過程,不會(huì)調(diào)用draw過程,也不會(huì)重新繪制任何View包括該調(diào)用者本身

onSizeChange(int w, int h, int oldw, int oldh)

onSizeChange是控件的大小發(fā)生變化的時(shí)候的調(diào)用,它的調(diào)用軌跡是layout->setFrame->sizeChange->onSizeChange。當(dāng)控件第一次布局時(shí)肯定會(huì)被調(diào)用到,我們覆寫該方法可以獲取到控件的大小。所以這個(gè)方法通常被用來在里面初始化跟控件大小相關(guān)的成員變量。

invalidate

invalidate使用的非常頻繁,它會(huì)觸發(fā)View的重新繪制,也就是繪制流程的draw過程,但不會(huì)調(diào)用測量和布局過程

postInvalidate

我們都知道Android的UI是單線程模型,只能在主線程更新UI,所以我們只能在主線程調(diào)用invalidate,如果想要在子線程更新ui,可以使用handler發(fā)送一個(gè)msg到主線程,然后在處理msg的時(shí)候去調(diào)用invalidate。另外,我們可以直接調(diào)用postInvalidate去在子線程更新UI,postInvalidate內(nèi)部實(shí)現(xiàn)也是使用handler來做發(fā)送msg到主線程然后調(diào)用invalidate。

setWillNotDraw

自定義ViewGroup通常是不會(huì)去繪制自己的,如果大家重寫ViewGroup里面的draw方法或者onDraw方法會(huì)發(fā)現(xiàn)它們根本就不會(huì)被調(diào)用到。但是如果給你的ViewGroup設(shè)置一個(gè)背景,就會(huì)發(fā)現(xiàn)draw方法和onDraw方法又都會(huì)走了。

我們知道ViewGroup本身是一個(gè)View,它的繪制是被其父容器發(fā)起的,具體的位置是在ViewGroup中的drawChild方法:

protected boolean drawChild(Canvas canvas, View child, long drawingTime) {
        return child.draw(canvas, this, drawingTime);
    }

注意這里的draw方法是帶三個(gè)參數(shù)的,與我們通常講的帶一個(gè)參數(shù)的draw方法不一樣。在View類中找到帶三個(gè)參數(shù)的draw方法,發(fā)現(xiàn)里面有這么一段代碼:

if (!hasDisplayList) {
                    // Fast path for layouts with no backgrounds
                    if ((mPrivateFlags & PFLAG_SKIP_DRAW) == PFLAG_SKIP_DRAW) {
                        mPrivateFlags &= ~PFLAG_DIRTY_MASK;
                        dispatchDraw(canvas);
                    } else {
                        draw(canvas);
                    }
                }

從這里我們可以看出一點(diǎn)端倪,通常一個(gè)ViewGroup默認(rèn)是會(huì)跳過繪制的,也即 (mPrivateFlags & PFLAG_SKIP_DRAW) == PFLAG_SKIP_DRAW 會(huì)返回一個(gè)true,那么會(huì)直接走dispatchDraw方法去畫它自己的孩子去了,并不會(huì)調(diào)用帶一個(gè)參數(shù)的 draw(canvas) ,但是當(dāng)這個(gè)ViewGroup有背景或者setWillNotDraw(false)是,就會(huì)走 draw(canvas) 方法。

所以如果我們自定義一個(gè)ViewGroup并且想要實(shí)現(xiàn)它本身的繪制的話,就可以給它設(shè)置一個(gè)背景或者調(diào)用 setWillNotDraw(false)

onAttachedToWindow

onAttachedToWindow是當(dāng)一個(gè)View綁定到window上時(shí)的調(diào)用,根據(jù)View類里面的對這個(gè)方法的注釋,onAttachedToWindow肯定會(huì)在onDraw方法之前調(diào)用。

在自定義控件里面,我們可以在onAttachedToWindow注冊一些廣播接收器,觀察者或者開啟一些任務(wù),大家可以參考TextClock的里面的實(shí)現(xiàn)。

onDetachedFromWindow

onDetachedFromWindow對應(yīng)于onAttachedToWindow,是當(dāng)一個(gè)View從window上移除時(shí)的一個(gè)調(diào)用。如果在onAttachedWindow里面注冊了一些監(jiān)聽,那么通常就要在onDetachedFromWindow里面反注冊。

ViewTreeObserver

ViewTreeObserver是視圖樹的觀察者,監(jiān)聽一些視圖樹的全局變化,這些全局變化包括整個(gè)視圖樹的布局,開始繪制,觸摸模式的變化等。我們不能直接初始化ViewTreeObserver的對象,需要通過getViewTreeObserver()去獲取。

ViewTreeObserver.OnGlobalLayoutListener

當(dāng)在一個(gè)視圖樹中全局布局發(fā)生改變或者視圖樹中的某個(gè)視圖的可視狀態(tài)發(fā)生改變的監(jiān)聽器,一般的使用姿勢是:

getViewTreeObserver().addOnGlobalLayoutListener(new OnGlobalLayoutListener() {

            @Override
            public void onGlobalLayout() {
                getViewTreeObserver().removeGlobalOnLayoutListener(this);
                //do something you like
                //for example, get view width or height height
            }
        });

ViewTreeObserver.OnPreDrawListener

當(dāng)一個(gè)視圖樹將要繪制時(shí)的監(jiān)聽器,一般的使用姿勢是:

getViewTreeObserver().addOnPreDrawListener(new OnPreDrawListener() {

            @Override
            public boolean onPreDraw() {
                //do something before draw
                //for example, request a new layout
                return true;
            }
        });

 

來自:http://www.jianshu.com/p/744550c02cf1

標(biāo)簽: isp 代碼

版權(quán)申明:本站文章部分自網(wǎng)絡(luò),如有侵權(quán),請聯(lián)系:west999com@outlook.com
特別注意:本站所有轉(zhuǎn)載文章言論不代表本站觀點(diǎn)!
本站所提供的圖片等素材,版權(quán)歸原作者所有,如需使用,請與原作者聯(lián)系。

上一篇:Python利用ctypes提高執(zhí)行速度

下一篇:2016最全面的編程語言技術(shù)開發(fā)視頻+源碼+技巧+軟件+面試全匯總