Android的UI界面都是由View和ViewGroup及其派生類組合而成的。canvas
其中,View是全部UI組件的基類,而 ViewGroup是容納這些組件的容器,其自己也是從View派生出來的.架構
View對象是Android平臺中用戶界面體現的基礎單位。框架
View類是它稱爲「widgets(工具)」的子類的基礎,它們提供了諸如文本輸入框和按鈕之類的UI對象的完整實現。工具
ViewGroup類一樣爲其被稱爲「Layouts(佈局)」的子類奠基了基礎,它們提供了象流式佈局、表格佈局以及相對佈局之類的佈局架構。佈局
通常來講,開發Android應用程序的UI界面都不會直接使用View和ViewGroup,而是使用這兩大基類的派生類。spa
View派生出的直接子類有:AnalogClock,ImageView,KeyboardView, ProgressBar,SurfaceView, TextView,ViewGroup,ViewStubxml
View派生出的間接子類有:AbsListView,AbsSeekBar, AbsSpinner, AbsoluteLayout, AdapterView<T extends Adapter>,AdapterViewAnimator, AdapterViewFlipper, AppWidgetHostView, AutoCompleteTextView,Button,CalendarView, CheckBox, CheckedTextView, Chronometer, CompoundButton,對象
ViewGroup派生出的直接子類有:AbsoluteLayout,AdapterView<T extends Adapter>,FragmentBreadCrumbs,FrameLayout, LinearLayout,RelativeLayout,SlidingDrawer遞歸
ViewGroup派生出的間接子類有:AbsListView,AbsSpinner, AdapterViewAnimator, AdapterViewFlipper, AppWidgetHostView, CalendarView, DatePicker, DialerFilter, ExpandableListView, Gallery, GestureOverlayView,GridView,HorizontalScrollView, ImageSwitcher,ListView,遊戲
這裏特別指出,ImageView是佈局具備圖片效果的UI經常使用的類,SurfaceView是用來進行遊戲開發的與通常View相比較爲特殊的很是重要的類,而AbsoluteLayout、 FrameLayout,LinearLayout, RelativeLayout這幾個ViewGroup的直接子類是Android UI佈局中最基本的佈局元素。
自定義控件(自定義View和ViewGroup)
給你們介紹下View和ViewGroup最重要的幾個方法——
protected void onDraw(Canvas canvas):View類中用於重繪的方法,這個方法是全部View、ViewGroup及其派生類都具備的方法,也是Android UI繪製最重要的方法。開發者可重載該方法,並在重載的方法內部基於參數canvas繪製本身的各類圖形、圖像效果。
protected void onLayout(boolean changed, int left, int top, int right, int bottom):View類中佈局發生改變時會調用的方法,這個方法是全部View、ViewGroup及其派生類都具備的方法,重載該類能夠在佈局發生改變時做定製處理,這在實現一些特效時很是有用。
protected void dispatchDraw(Canvas canvas):ViewGroup類及其派生類具備的方法,這個方法主要用於控制子View的繪製分發,重載該方法可改變子View的繪製,進而實現一些複雜的視效,典型的例子可參見Launcher模塊Workspace的dispatchDraw重載。
protected boolean drawChild(Canvas canvas, View child, long drawingTime)):ViewGroup類及其派生類具備的方法,這個方法直接控制繪製某局具體的子view,重載該方法可控制具體某個具體子View。
addView方法這個方法是用來向View容器中添加組件用的。咱們可使用這個方法向這個ViewGroup中添加組件。
getChildAt方法 這個方法用來返回指定位置的View。
注意:ViewGroup中的View是從0開始計數的。
View在屏幕上顯示出來要先通過measure(計算)和layout(佈局).
onMeasure(int, int) View會調用此方法,來確認本身及全部子對象的大小
onLayout(boolean, int, int, int, int, int, int) 當View要爲全部子對象分配大小和位置時,調用此方法
onSizeChanged(int, int, int, int) 當View大小改變時,調用此方法
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec)介紹:
onMeasure傳入的widthMeasureSpec和heightMeasureSpec不是通常的尺寸數值,而是將模式和尺寸組合在一塊兒的數值。
通常是根據xml文件中定義獲得的,咱們能夠根據這2個參數知道模式和size。
咱們須要經過int mode = MeasureSpec.getMode(widthMeasureSpec)獲得模式,
用int size = MeasureSpec.getSize(widthMeasureSpec)獲得尺寸。
mode共有三種狀況,取值分別爲MeasureSpec.UNSPECIFIED, MeasureSpec.EXACTLY, MeasureSpec.AT_MOST。
對應關係:
-2147483648----xml文件中的wrap_content----MeasureSpec.AT_MOST
1073741824----xml文件中的fill_parent-----MeasureSpec.EXACTLY
0-----MeasureSpec.UNSPECIFIED
通常ViewGroup,咱們是這樣實現的:
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
final int width = MeasureSpec.getSize(widthMeasureSpec);
final int widthMode = MeasureSpec.getMode(widthMeasureSpec);
final int count = getChildCount();
for (int i = 0; i < count; i++) {
//其實,對於咱們本身寫的應用來講,最好的辦法是去掉框架裏的該方法,直接調用view.measure(),以下:
//通常咱們設定ViewGroup的XML佈局是wrap_content,這樣2個參數就是-2147483648,那麼咱們以下調用就是
//讓子組件本身適配大小
getChildAt(i).measure(widthMeasureSpec, heightMeasureSpec);
//還有下面的方法
//整個measure()過程就是個遞歸過程
//該方法只是一個過濾器,最後會調用measure()過程 ;或者 measureChild(child , h, i)方法
//measureChildWithMargins(getChildAt(i). , h, i) ;
}
scrollTo(mControl.getCurScreen() * width, 0);
}
固然咱們也能夠調用setMeasuredDimension(h , l) ; 來設定ViewGroup的大小。
至於View的onMeasure實現,咱們其實通常都不覆寫該方法,覆寫的話也簡單,根據須要,和上面同樣,根據傳入的2個參數獲取當前的模式和大小。
固然咱們也能夠本身計算大小,調用setMeasuredDimension設定。
ViewGroup的onLayout實現:
protected void onLayout(boolean changed, int l, int t, int r, int b) {
int childLeft = 0;
// 獲取全部的子View的個數
final int childCount = getChildCount();
for (int i = 0; i < childCount; i++) {
final View childView = getChildAt(i);
final int childWidth = childView.getMeasuredWidth();
final int childHeight = childView.getMeasuredHeight();
childView.layout(childLeft, 0, childLeft + childWidth,childHeight);
// 下一個VIew的左邊左邊+一個
childLeft += childWidth;
}
}
實際上很簡單,就是調用layout方法來設定View在畫布上的位置,能夠超出屏幕寬度,而後咱們能夠滾動顯示。固然咱們也能夠調用layout傳入相關座標來設定View顯示位置(已驗證OK)